Commit 58cbf3d3 authored by Toby Dickenson's avatar Toby Dickenson

fixed line endings

parent 2eede254
# #
# XML-RPC CLIENT LIBRARY # XML-RPC CLIENT LIBRARY
# $Id$ # $Id$
# #
# an XML-RPC client interface for Python. # an XML-RPC client interface for Python.
# #
# the marshalling and response parser code can also be used to # the marshalling and response parser code can also be used to
# implement XML-RPC servers. # implement XML-RPC servers.
# #
# Notes: # Notes:
# this version uses the sgmlop XML parser, if installed. this is # this version uses the sgmlop XML parser, if installed. this is
# typically 10-15x faster than using Python's standard XML parser. # typically 10-15x faster than using Python's standard XML parser.
# #
# you can get the sgmlop distribution from: # you can get the sgmlop distribution from:
# #
# http://www.pythonware.com/products/xml/sgmlop.htm # http://www.pythonware.com/products/xml/sgmlop.htm
# #
# this version is designed to work with Python 1.5.2 or newer. # this version is designed to work with Python 1.5.2 or newer.
# unicode encoding support requires at least Python 1.6. # unicode encoding support requires at least Python 1.6.
# experimental HTTPS requires Python 2.0 built with SSL sockets. # experimental HTTPS requires Python 2.0 built with SSL sockets.
# #
# History: # History:
# 1999-01-14 fl Created # 1999-01-14 fl Created
# 1999-01-15 fl Changed dateTime to use localtime # 1999-01-15 fl Changed dateTime to use localtime
# 1999-01-16 fl Added Binary/base64 element, default to RPC2 service # 1999-01-16 fl Added Binary/base64 element, default to RPC2 service
# 1999-01-19 fl Fixed array data element (from Skip Montanaro) # 1999-01-19 fl Fixed array data element (from Skip Montanaro)
# 1999-01-21 fl Fixed dateTime constructor, etc. # 1999-01-21 fl Fixed dateTime constructor, etc.
# 1999-02-02 fl Added fault handling, handle empty sequences, etc. # 1999-02-02 fl Added fault handling, handle empty sequences, etc.
# 1999-02-10 fl Fixed problem with empty responses (from Skip Montanaro) # 1999-02-10 fl Fixed problem with empty responses (from Skip Montanaro)
# 1999-06-20 fl Speed improvements, pluggable parsers/transports (0.9.8) # 1999-06-20 fl Speed improvements, pluggable parsers/transports (0.9.8)
# 2000-11-28 fl Changed boolean to check the truth value of its argument # 2000-11-28 fl Changed boolean to check the truth value of its argument
# 2001-02-24 fl Added encoding/Unicode/SafeTransport patches # 2001-02-24 fl Added encoding/Unicode/SafeTransport patches
# 2001-02-26 fl Added compare support to wrappers (0.9.9) # 2001-02-26 fl Added compare support to wrappers (0.9.9)
# #
# Copyright (c) 1999-2001 by Secret Labs AB. # Copyright (c) 1999-2001 by Secret Labs AB.
# Copyright (c) 1999-2001 by Fredrik Lundh. # Copyright (c) 1999-2001 by Fredrik Lundh.
# #
# fredrik@pythonware.com # fredrik@pythonware.com
# http://www.pythonware.com # http://www.pythonware.com
# #
# -------------------------------------------------------------------- # --------------------------------------------------------------------
# The XML-RPC client interface is # The XML-RPC client interface is
# #
# Copyright (c) 1999-2001 by Secret Labs AB # Copyright (c) 1999-2001 by Secret Labs AB
# Copyright (c) 1999-2001 by Fredrik Lundh # Copyright (c) 1999-2001 by Fredrik Lundh
# #
# By obtaining, using, and/or copying this software and/or its # By obtaining, using, and/or copying this software and/or its
# associated documentation, you agree that you have read, understood, # associated documentation, you agree that you have read, understood,
# and will comply with the following terms and conditions: # and will comply with the following terms and conditions:
# #
# Permission to use, copy, modify, and distribute this software and # Permission to use, copy, modify, and distribute this software and
# its associated documentation for any purpose and without fee is # its associated documentation for any purpose and without fee is
# hereby granted, provided that the above copyright notice appears in # hereby granted, provided that the above copyright notice appears in
# all copies, and that both that copyright notice and this permission # all copies, and that both that copyright notice and this permission
# notice appear in supporting documentation, and that the name of # notice appear in supporting documentation, and that the name of
# Secret Labs AB or the author not be used in advertising or publicity # Secret Labs AB or the author not be used in advertising or publicity
# pertaining to distribution of the software without specific, written # pertaining to distribution of the software without specific, written
# prior permission. # prior permission.
# #
# SECRET LABS AB AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH REGARD # SECRET LABS AB AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH REGARD
# TO THIS SOFTWARE, INCLUDING ALL IMPLIED WARRANTIES OF MERCHANT- # TO THIS SOFTWARE, INCLUDING ALL IMPLIED WARRANTIES OF MERCHANT-
# ABILITY AND FITNESS. IN NO EVENT SHALL SECRET LABS AB OR THE AUTHOR # ABILITY AND FITNESS. IN NO EVENT SHALL SECRET LABS AB OR THE AUTHOR
# BE LIABLE FOR ANY SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY # BE LIABLE FOR ANY SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY
# DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, # DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS,
# WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS # WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS
# ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE # ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE
# OF THIS SOFTWARE. # OF THIS SOFTWARE.
# -------------------------------------------------------------------- # --------------------------------------------------------------------
# #
# things to fix before 1.0 final: # things to fix before 1.0 final:
# TODO: unicode marshalling -DONE # TODO: unicode marshalling -DONE
# TODO: ascii-compatible encoding support -DONE # TODO: ascii-compatible encoding support -DONE
# TODO: safe transport -DONE (but mostly untested) # TODO: safe transport -DONE (but mostly untested)
# TODO: sgmlop memory leak -DONE # TODO: sgmlop memory leak -DONE
# TODO: sgmlop xml parsing -DONE # TODO: sgmlop xml parsing -DONE
# TODO: support unicode method names -DONE # TODO: support unicode method names -DONE
# TODO: update selftest -DONE # TODO: update selftest -DONE
# TODO: add docstrings -DONE # TODO: add docstrings -DONE
# TODO: clean up parser encoding (trust the parser) -DONE # TODO: clean up parser encoding (trust the parser) -DONE
# TODO: fix host tuple handling in the server constructor # TODO: fix host tuple handling in the server constructor
# TODO: let transport verify schemes # TODO: let transport verify schemes
# TODO: update documentation # TODO: update documentation
# TODO: authentication plugins # TODO: authentication plugins
# TODO: memo problem (see HP's mail) # TODO: memo problem (see HP's mail)
import re, string, time, operator import re, string, time, operator
import urllib, xmllib import urllib, xmllib
from types import * from types import *
from cgi import escape from cgi import escape
try: try:
import sgmlop import sgmlop
if not hasattr(sgmlop, "XMLParser"): if not hasattr(sgmlop, "XMLParser"):
raise ImportError raise ImportError
except ImportError: except ImportError:
sgmlop = None # accelerator not available sgmlop = None # accelerator not available
try: try:
unicode unicode
except NameError: except NameError:
unicode = None # unicode support not available unicode = None # unicode support not available
def _decode(data, encoding, is8bit=re.compile("[\x80-\xff]").search): def _decode(data, encoding, is8bit=re.compile("[\x80-\xff]").search):
# decode non-ascii string (if possible) # decode non-ascii string (if possible)
if unicode and is8bit(data): if unicode and is8bit(data):
data = unicode(data, encoding) data = unicode(data, encoding)
return data return data
__version__ = "0.9.9" __version__ = "0.9.9"
# -------------------------------------------------------------------- # --------------------------------------------------------------------
# Exceptions # Exceptions
class Error: class Error:
# base class for client errors # base class for client errors
pass pass
class ProtocolError(Error): class ProtocolError(Error):
# indicates an HTTP protocol error # indicates an HTTP protocol error
def __init__(self, url, errcode, errmsg, headers): def __init__(self, url, errcode, errmsg, headers):
self.url = url self.url = url
self.errcode = errcode self.errcode = errcode
self.errmsg = errmsg self.errmsg = errmsg
self.headers = headers self.headers = headers
def __repr__(self): def __repr__(self):
return ( return (
"<ProtocolError for %s: %s %s>" % "<ProtocolError for %s: %s %s>" %
(self.url, self.errcode, self.errmsg) (self.url, self.errcode, self.errmsg)
) )
class ResponseError(Error): class ResponseError(Error):
# indicates a broken response package # indicates a broken response package
pass pass
class Fault(Error): class Fault(Error):
# indicates a XML-RPC fault package # indicates a XML-RPC fault package
def __init__(self, faultCode, faultString, **extra): def __init__(self, faultCode, faultString, **extra):
self.faultCode = faultCode self.faultCode = faultCode
self.faultString = faultString self.faultString = faultString
def __repr__(self): def __repr__(self):
return ( return (
"<Fault %s: %s>" % "<Fault %s: %s>" %
(self.faultCode, repr(self.faultString)) (self.faultCode, repr(self.faultString))
) )
# -------------------------------------------------------------------- # --------------------------------------------------------------------
# Special values # Special values
# boolean wrapper # boolean wrapper
# use True or False to generate a "boolean" XML-RPC value # use True or False to generate a "boolean" XML-RPC value
class Boolean: class Boolean:
def __init__(self, value = 0): def __init__(self, value = 0):
self.value = operator.truth(value) self.value = operator.truth(value)
def encode(self, out): def encode(self, out):
out.write("<value><boolean>%d</boolean></value>\n" % self.value) out.write("<value><boolean>%d</boolean></value>\n" % self.value)
def __cmp__(self, other): def __cmp__(self, other):
if isinstance(other, Boolean): if isinstance(other, Boolean):
other = other.value other = other.value
return cmp(self.value, other) return cmp(self.value, other)
def __repr__(self): def __repr__(self):
if self.value: if self.value:
return "<Boolean True at %x>" % id(self) return "<Boolean True at %x>" % id(self)
else: else:
return "<Boolean False at %x>" % id(self) return "<Boolean False at %x>" % id(self)
def __int__(self): def __int__(self):
return self.value return self.value
def __nonzero__(self): def __nonzero__(self):
return self.value return self.value
True, False = Boolean(1), Boolean(0) True, False = Boolean(1), Boolean(0)
def boolean(value, truefalse=(False, True)): def boolean(value, truefalse=(False, True)):
# convert any Python value to XML-RPC boolean # convert any Python value to XML-RPC boolean
return truefalse[operator.truth(value)] return truefalse[operator.truth(value)]
# #
# dateTime wrapper # dateTime wrapper
# wrap your iso8601 string or time tuple or localtime integer value # wrap your iso8601 string or time tuple or localtime integer value
# in this class to generate a "dateTime.iso8601" XML-RPC value # in this class to generate a "dateTime.iso8601" XML-RPC value
class DateTime: class DateTime:
def __init__(self, value=0): def __init__(self, value=0):
t = type(value) t = type(value)
if not isinstance(t, StringType): if not isinstance(t, StringType):
if not isinstance(t, TupleType): if not isinstance(t, TupleType):
if value == 0: if value == 0:
value = time.time() value = time.time()
value = time.localtime(value) value = time.localtime(value)
value = time.strftime("%Y%m%dT%H:%M:%S", value) value = time.strftime("%Y%m%dT%H:%M:%S", value)
self.value = value self.value = value
def __cmp__(self, other): def __cmp__(self, other):
if isinstance(other, DateTime): if isinstance(other, DateTime):
other = other.value other = other.value
return cmp(self.value, other) return cmp(self.value, other)
def __repr__(self): def __repr__(self):
return "<DateTime %s at %x>" % (self.value, id(self)) return "<DateTime %s at %x>" % (self.value, id(self))
def decode(self, data): def decode(self, data):
self.value = string.strip(data) self.value = string.strip(data)
def encode(self, out): def encode(self, out):
out.write("<value><dateTime.iso8601>") out.write("<value><dateTime.iso8601>")
out.write(self.value) out.write(self.value)
out.write("</dateTime.iso8601></value>\n") out.write("</dateTime.iso8601></value>\n")
# #
# binary data wrapper # binary data wrapper
class Binary: class Binary:
def __init__(self, data=None): def __init__(self, data=None):
self.data = data self.data = data
def __cmp__(self, other): def __cmp__(self, other):
if isinstance(other, Binary): if isinstance(other, Binary):
other = other.data other = other.data
return cmp(self.data, other) return cmp(self.data, other)
def decode(self, data): def decode(self, data):
import base64 import base64
self.data = base64.decodestring(data) self.data = base64.decodestring(data)
def encode(self, out): def encode(self, out):
import base64, StringIO import base64, StringIO
out.write("<value><base64>\n") out.write("<value><base64>\n")
base64.encode(StringIO.StringIO(self.data), out) base64.encode(StringIO.StringIO(self.data), out)
out.write("</base64></value>\n") out.write("</base64></value>\n")
WRAPPERS = DateTime, Binary, Boolean WRAPPERS = DateTime, Binary, Boolean
# -------------------------------------------------------------------- # --------------------------------------------------------------------
# XML parsers # XML parsers
if sgmlop: if sgmlop:
class FastParser: class FastParser:
# sgmlop based XML parser. this is typically 15x faster # sgmlop based XML parser. this is typically 15x faster
# than SlowParser... # than SlowParser...
def __init__(self, target): def __init__(self, target):
# setup callbacks # setup callbacks
self.finish_starttag = target.start self.finish_starttag = target.start
self.finish_endtag = target.end self.finish_endtag = target.end
self.handle_data = target.data self.handle_data = target.data
self.handle_xml = target.xml self.handle_xml = target.xml
# activate parser # activate parser
self.parser = sgmlop.XMLParser() self.parser = sgmlop.XMLParser()
self.parser.register(self) self.parser.register(self)
self.feed = self.parser.feed self.feed = self.parser.feed
self.entity = { self.entity = {
"amp": "&", "gt": ">", "lt": "<", "amp": "&", "gt": ">", "lt": "<",
"apos": "'", "quot": '"' "apos": "'", "quot": '"'
} }
def close(self): def close(self):
try: try:
self.parser.close() self.parser.close()
finally: finally:
self.parser = self.feed = None # nuke circular reference self.parser = self.feed = None # nuke circular reference
def handle_proc(self, tag, attr): def handle_proc(self, tag, attr):
m = re.search("encoding\s*=\s*['\"]([^\"']+)[\"']", attr) m = re.search("encoding\s*=\s*['\"]([^\"']+)[\"']", attr)
if m: if m:
self.handle_xml(m.group(1), 1) self.handle_xml(m.group(1), 1)
def handle_entityref(self, entity): def handle_entityref(self, entity):
# <string> entity # <string> entity
try: try:
self.handle_data(self.entity[entity]) self.handle_data(self.entity[entity])
except KeyError: except KeyError:
self.handle_data("&%s;" % entity) self.handle_data("&%s;" % entity)
else: else:
FastParser = None FastParser = None
class SlowParser(xmllib.XMLParser): class SlowParser(xmllib.XMLParser):
# slow but safe standard parser, based on the XML parser in # slow but safe standard parser, based on the XML parser in
# Python's standard library # Python's standard library
def __init__(self, target): def __init__(self, target):
self.handle_xml = target.xml self.handle_xml = target.xml
self.unknown_starttag = target.start self.unknown_starttag = target.start
self.handle_data = target.data self.handle_data = target.data
self.unknown_endtag = target.end self.unknown_endtag = target.end
xmllib.XMLParser.__init__(self) xmllib.XMLParser.__init__(self)
# -------------------------------------------------------------------- # --------------------------------------------------------------------
# XML-RPC marshalling and unmarshalling code # XML-RPC marshalling and unmarshalling code
class Marshaller: class Marshaller:
"""Generate an XML-RPC params chunk from a Python data structure""" """Generate an XML-RPC params chunk from a Python data structure"""
# USAGE: create a marshaller instance for each set of parameters, # USAGE: create a marshaller instance for each set of parameters,
# and use "dumps" to convert your data (represented as a tuple) to # and use "dumps" to convert your data (represented as a tuple) to
# a XML-RPC params chunk. to write a fault response, pass a Fault # a XML-RPC params chunk. to write a fault response, pass a Fault
# instance instead. you may prefer to use the "dumps" convenience # instance instead. you may prefer to use the "dumps" convenience
# function for this purpose (see below). # function for this purpose (see below).
# by the way, if you don't understand what's going on in here, # by the way, if you don't understand what's going on in here,
# that's perfectly ok. # that's perfectly ok.
def __init__(self, encoding=None): def __init__(self, encoding=None):
self.memo = {} self.memo = {}
self.data = None self.data = None
self.encoding = encoding self.encoding = encoding
dispatch = {} dispatch = {}
def dumps(self, values): def dumps(self, values):
self.__out = [] self.__out = []
self.write = write = self.__out.append self.write = write = self.__out.append
if isinstance(values, Fault): if isinstance(values, Fault):
# fault instance # fault instance
write("<fault>\n") write("<fault>\n")
self.__dump(vars(values)) self.__dump(vars(values))
write("</fault>\n") write("</fault>\n")
else: else:
# parameter block # parameter block
write("<params>\n") write("<params>\n")
for v in values: for v in values:
write("<param>\n") write("<param>\n")
self.__dump(v) self.__dump(v)
write("</param>\n") write("</param>\n")
write("</params>\n") write("</params>\n")
result = string.join(self.__out, "") result = string.join(self.__out, "")
del self.__out, self.write # don't need this any more del self.__out, self.write # don't need this any more
return result return result
def __dump(self, value): def __dump(self, value):
try: try:
f = self.dispatch[type(value)] f = self.dispatch[type(value)]
except KeyError: except KeyError:
raise TypeError, "cannot marshal %s objects" % type(value) raise TypeError, "cannot marshal %s objects" % type(value)
else: else:
f(self, value) f(self, value)
def dump_int(self, value): def dump_int(self, value):
self.write("<value><int>%s</int></value>\n" % value) self.write("<value><int>%s</int></value>\n" % value)
dispatch[IntType] = dump_int dispatch[IntType] = dump_int
def dump_double(self, value): def dump_double(self, value):
self.write("<value><double>%s</double></value>\n" % value) self.write("<value><double>%s</double></value>\n" % value)
dispatch[FloatType] = dump_double dispatch[FloatType] = dump_double
def dump_string(self, value): def dump_string(self, value):
self.write("<value><string>%s</string></value>\n" % escape(value)) self.write("<value><string>%s</string></value>\n" % escape(value))
dispatch[StringType] = dump_string dispatch[StringType] = dump_string
if unicode: if unicode:
def dump_unicode(self, value): def dump_unicode(self, value):
value = value.encode(self.encoding) value = value.encode(self.encoding)
self.write("<value><string>%s</string></value>\n" % escape(value)) self.write("<value><string>%s</string></value>\n" % escape(value))
dispatch[UnicodeType] = dump_unicode dispatch[UnicodeType] = dump_unicode
# Zope-specific extension; xmlrpc doesnt have an equivalent of # Zope-specific extension; xmlrpc doesnt have an equivalent of
# Python's None. the standard xmlrpclib raises an exception, # Python's None. the standard xmlrpclib raises an exception,
# but marshalling it as a zero is more convenient. # but marshalling it as a zero is more convenient.
def dump_none(self, value): def dump_none(self, value):
self.write("<value><int>0</int></value>\n") self.write("<value><int>0</int></value>\n")
dispatch[NoneType] = dump_none dispatch[NoneType] = dump_none
def container(self, value): def container(self, value):
if value: if value:
i = id(value) i = id(value)
if self.memo.has_key(i): if self.memo.has_key(i):
raise TypeError, "cannot marshal recursive data structures" raise TypeError, "cannot marshal recursive data structures"
self.memo[i] = None self.memo[i] = None
def dump_array(self, value): def dump_array(self, value):
self.container(value) self.container(value)
write = self.write write = self.write
write("<value><array><data>\n") write("<value><array><data>\n")
for v in value: for v in value:
self.__dump(v) self.__dump(v)
write("</data></array></value>\n") write("</data></array></value>\n")
dispatch[TupleType] = dump_array dispatch[TupleType] = dump_array
dispatch[ListType] = dump_array dispatch[ListType] = dump_array
def dump_struct(self, value): def dump_struct(self, value):
self.container(value) self.container(value)
write = self.write write = self.write
write("<value><struct>\n") write("<value><struct>\n")
for k, v in value.items(): for k, v in value.items():
write("<member>\n") write("<member>\n")
if type(k) is not StringType: if type(k) is not StringType:
raise TypeError, "dictionary key must be string" raise TypeError, "dictionary key must be string"
write("<name>%s</name>\n" % escape(k)) write("<name>%s</name>\n" % escape(k))
self.__dump(v) self.__dump(v)
write("</member>\n") write("</member>\n")
write("</struct></value>\n") write("</struct></value>\n")
dispatch[DictType] = dump_struct dispatch[DictType] = dump_struct
def dump_instance(self, value): def dump_instance(self, value):
# check for special wrappers # check for special wrappers
if value.__class__ in WRAPPERS: if value.__class__ in WRAPPERS:
value.encode(self) value.encode(self)
else: else:
# store instance attributes as a struct (really?) # store instance attributes as a struct (really?)
self.dump_struct(value.__dict__) self.dump_struct(value.__dict__)
dispatch[InstanceType] = dump_instance dispatch[InstanceType] = dump_instance
class Unmarshaller: class Unmarshaller:
# unmarshal an XML-RPC response, based on incoming XML event # unmarshal an XML-RPC response, based on incoming XML event
# messages (start, data, end). call close to get the resulting # messages (start, data, end). call close to get the resulting
# data structure # data structure
# note that this reader is fairly tolerant, and gladly accepts # note that this reader is fairly tolerant, and gladly accepts
# bogus XML-RPC data without complaining (but not bogus XML). # bogus XML-RPC data without complaining (but not bogus XML).
# and again, if you don't understand what's going on in here, # and again, if you don't understand what's going on in here,
# that's perfectly ok. # that's perfectly ok.
def __init__(self): def __init__(self):
self._type = None self._type = None
self._stack = [] self._stack = []
self._marks = [] self._marks = []
self._data = [] self._data = []
self._methodname = None self._methodname = None
self._encoding = "utf-8" self._encoding = "utf-8"
self.append = self._stack.append self.append = self._stack.append
def close(self): def close(self):
# return response tuple and target method # return response tuple and target method
if self._type is None or self._marks: if self._type is None or self._marks:
raise ResponseError() raise ResponseError()
if self._type == "fault": if self._type == "fault":
raise apply(Fault, (), self._stack[0]) raise apply(Fault, (), self._stack[0])
return tuple(self._stack) return tuple(self._stack)
def getmethodname(self): def getmethodname(self):
return self._methodname return self._methodname
# #
# event handlers # event handlers
def xml(self, encoding, standalone): def xml(self, encoding, standalone):
self._encoding = encoding or "utf-8" self._encoding = encoding or "utf-8"
# FIXME: assert standalone == 1 ??? # FIXME: assert standalone == 1 ???
def start(self, tag, attrs): def start(self, tag, attrs):
# prepare to handle this element # prepare to handle this element
if tag in ("array", "struct"): if tag in ("array", "struct"):
self._marks.append(len(self._stack)) self._marks.append(len(self._stack))
self._data = [] self._data = []
self._value = (tag == "value") self._value = (tag == "value")
def data(self, text): def data(self, text):
self._data.append(text) self._data.append(text)
dispatch = {} dispatch = {}
def end(self, tag): def end(self, tag):
# call the appropriate end tag handler # call the appropriate end tag handler
try: try:
f = self.dispatch[tag] f = self.dispatch[tag]
except KeyError: except KeyError:
pass # unknown tag ? pass # unknown tag ?
else: else:
return f(self) return f(self)
# #
# element decoders # element decoders
def end_boolean(self, join=string.join): def end_boolean(self, join=string.join):
value = join(self._data, "") value = join(self._data, "")
if value == "0": if value == "0":
self.append(False) self.append(False)
elif value == "1": elif value == "1":
self.append(True) self.append(True)
else: else:
raise TypeError, "bad boolean value" raise TypeError, "bad boolean value"
self._value = 0 self._value = 0
dispatch["boolean"] = end_boolean dispatch["boolean"] = end_boolean
def end_int(self, join=string.join): def end_int(self, join=string.join):
self.append(int(join(self._data, ""))) self.append(int(join(self._data, "")))
self._value = 0 self._value = 0
dispatch["i4"] = end_int dispatch["i4"] = end_int
dispatch["int"] = end_int dispatch["int"] = end_int
def end_double(self, join=string.join): def end_double(self, join=string.join):
self.append(float(join(self._data, ""))) self.append(float(join(self._data, "")))
self._value = 0 self._value = 0
dispatch["double"] = end_double dispatch["double"] = end_double
def end_string(self, join=string.join): def end_string(self, join=string.join):
data = join(self._data, "") data = join(self._data, "")
if self._encoding: if self._encoding:
data = _decode(data, self._encoding) data = _decode(data, self._encoding)
self.append(data) self.append(data)
self._value = 0 self._value = 0
dispatch["string"] = end_string dispatch["string"] = end_string
dispatch["name"] = end_string # struct keys are always strings dispatch["name"] = end_string # struct keys are always strings
def end_array(self): def end_array(self):
mark = self._marks[-1] mark = self._marks[-1]
del self._marks[-1] del self._marks[-1]
# map arrays to Python lists # map arrays to Python lists
self._stack[mark:] = [self._stack[mark:]] self._stack[mark:] = [self._stack[mark:]]
self._value = 0 self._value = 0
dispatch["array"] = end_array dispatch["array"] = end_array
def end_struct(self): def end_struct(self):
mark = self._marks[-1] mark = self._marks[-1]
del self._marks[-1] del self._marks[-1]
# map structs to Python dictionaries # map structs to Python dictionaries
dict = {} dict = {}
items = self._stack[mark:] items = self._stack[mark:]
for i in range(0, len(items), 2): for i in range(0, len(items), 2):
dict[items[i]] = items[i+1] dict[items[i]] = items[i+1]
self._stack[mark:] = [dict] self._stack[mark:] = [dict]
self._value = 0 self._value = 0
dispatch["struct"] = end_struct dispatch["struct"] = end_struct
def end_base64(self, join=string.join): def end_base64(self, join=string.join):
value = Binary() value = Binary()
value.decode(join(self._data, "")) value.decode(join(self._data, ""))
self.append(value) self.append(value)
self._value = 0 self._value = 0
dispatch["base64"] = end_base64 dispatch["base64"] = end_base64
def end_dateTime(self, join=string.join): def end_dateTime(self, join=string.join):
value = DateTime() value = DateTime()
value.decode(join(self._data, "")) value.decode(join(self._data, ""))
self.append(value) self.append(value)
dispatch["dateTime.iso8601"] = end_dateTime dispatch["dateTime.iso8601"] = end_dateTime
def end_value(self): def end_value(self):
# if we stumble upon an value element with no internal # if we stumble upon an value element with no internal
# elements, treat it as a string element # elements, treat it as a string element
if self._value: if self._value:
self.end_string() self.end_string()
dispatch["value"] = end_value dispatch["value"] = end_value
def end_params(self): def end_params(self):
self._type = "params" self._type = "params"
dispatch["params"] = end_params dispatch["params"] = end_params
def end_fault(self): def end_fault(self):
self._type = "fault" self._type = "fault"
dispatch["fault"] = end_fault dispatch["fault"] = end_fault
def end_methodName(self, join=string.join): def end_methodName(self, join=string.join):
data = join(self._data, "") data = join(self._data, "")
if self._encoding: if self._encoding:
data = _decode(data, self._encoding) data = _decode(data, self._encoding)
self._methodname = data self._methodname = data
dispatch["methodName"] = end_methodName dispatch["methodName"] = end_methodName
# -------------------------------------------------------------------- # --------------------------------------------------------------------
# convenience functions # convenience functions
def getparser(): def getparser():
"""getparser() -> parser, unmarshaller """getparser() -> parser, unmarshaller
Create an instance of the fastest available parser, and attach Create an instance of the fastest available parser, and attach
it to an unmarshalling object. Return both objects. it to an unmarshalling object. Return both objects.
""" """
target = Unmarshaller() target = Unmarshaller()
if FastParser: if FastParser:
return FastParser(target), target return FastParser(target), target
return SlowParser(target), target return SlowParser(target), target
def dumps(params, methodname=None, methodresponse=None, encoding=None): def dumps(params, methodname=None, methodresponse=None, encoding=None):
"""data [,options] -> marshalled data """data [,options] -> marshalled data
Convert a tuple or a fault object to an XML-RPC request (or Convert a tuple or a fault object to an XML-RPC request (or
response, if the methodsresponse option is used). response, if the methodsresponse option is used).
In addition to the data object, the following options can be In addition to the data object, the following options can be
given as keyword arguments: given as keyword arguments:
methodname: the method name for a methodCall packet methodname: the method name for a methodCall packet
methodresponse: true to create a methodResponse packet methodresponse: true to create a methodResponse packet
encoding: the packet encoding (default is UTF-8) encoding: the packet encoding (default is UTF-8)
All 8-bit strings in the data structure are assumed to use the All 8-bit strings in the data structure are assumed to use the
packet encoding. Unicode strings are automatically converted, packet encoding. Unicode strings are automatically converted,
as necessary. as necessary.
""" """
assert type(params) == TupleType or isinstance(params, Fault),\ assert type(params) == TupleType or isinstance(params, Fault),\
"argument must be tuple or Fault instance" "argument must be tuple or Fault instance"
if not encoding: if not encoding:
encoding = "utf-8" encoding = "utf-8"
m = Marshaller(encoding) m = Marshaller(encoding)
data = m.dumps(params) data = m.dumps(params)
if encoding != "utf-8": if encoding != "utf-8":
xmlheader = "<?xml version='1.0' encoding=%s?>\n" % repr(encoding) xmlheader = "<?xml version='1.0' encoding=%s?>\n" % repr(encoding)
else: else:
xmlheader = "<?xml version='1.0'?>\n" # utf-8 is default xmlheader = "<?xml version='1.0'?>\n" # utf-8 is default
# standard XML-RPC wrappings # standard XML-RPC wrappings
if methodname: if methodname:
# a method call # a method call
if not isinstance(methodname, StringType): if not isinstance(methodname, StringType):
methodname = methodname.encode(encoding) methodname = methodname.encode(encoding)
data = ( data = (
xmlheader, xmlheader,
"<methodCall>\n" "<methodCall>\n"
"<methodName>", methodname, "</methodName>\n", "<methodName>", methodname, "</methodName>\n",
data, data,
"</methodCall>\n" "</methodCall>\n"
) )
elif methodresponse or isinstance(params, Fault): elif methodresponse or isinstance(params, Fault):
# a method response # a method response
data = ( data = (
xmlheader, xmlheader,
"<methodResponse>\n", "<methodResponse>\n",
data, data,
"</methodResponse>\n" "</methodResponse>\n"
) )
else: else:
return data # return as is return data # return as is
return string.join(data, "") return string.join(data, "")
def loads(data): def loads(data):
"""data -> unmarshalled data, method name """data -> unmarshalled data, method name
Convert an XML-RPC packet to unmarshalled data plus a method Convert an XML-RPC packet to unmarshalled data plus a method
name (None if not present). name (None if not present).
If the XML-RPC packet represents a fault condition, this function If the XML-RPC packet represents a fault condition, this function
raises a Fault exception. raises a Fault exception.
""" """
p, u = getparser() p, u = getparser()
p.feed(data) p.feed(data)
p.close() p.close()
return u.close(), u.getmethodname() return u.close(), u.getmethodname()
# -------------------------------------------------------------------- # --------------------------------------------------------------------
# request dispatcher # request dispatcher
class _Method: class _Method:
# some magic to bind an XML-RPC method to an RPC server. # some magic to bind an XML-RPC method to an RPC server.
# supports "nested" methods (e.g. examples.getStateName) # supports "nested" methods (e.g. examples.getStateName)
def __init__(self, send, name): def __init__(self, send, name):
self.__send = send self.__send = send
self.__name = name self.__name = name
def __getattr__(self, name): def __getattr__(self, name):
return _Method(self.__send, "%s.%s" % (self.__name, name)) return _Method(self.__send, "%s.%s" % (self.__name, name))
def __call__(self, *args): def __call__(self, *args):
return self.__send(self.__name, args) return self.__send(self.__name, args)
class Transport: class Transport:
"""Handles an HTTP transaction to an XML-RPC server""" """Handles an HTTP transaction to an XML-RPC server"""
# client identifier (may be overridden) # client identifier (may be overridden)
user_agent = "xmlrpclib.py/%s (by www.pythonware.com)" % __version__ user_agent = "xmlrpclib.py/%s (by www.pythonware.com)" % __version__
def request(self, host, handler, request_body, verbose=0): def request(self, host, handler, request_body, verbose=0):
# issue XML-RPC request # issue XML-RPC request
h = self.make_connection(host) h = self.make_connection(host)
if verbose: if verbose:
h.set_debuglevel(1) h.set_debuglevel(1)
self.send_request(h, handler, request_body) self.send_request(h, handler, request_body)
self.send_host(h, host) self.send_host(h, host)
self.send_user_agent(h) self.send_user_agent(h)
self.send_content(h, request_body) self.send_content(h, request_body)
errcode, errmsg, headers = h.getreply() errcode, errmsg, headers = h.getreply()
if errcode != 200: if errcode != 200:
raise ProtocolError( raise ProtocolError(
host + handler, host + handler,
errcode, errmsg, errcode, errmsg,
headers headers
) )
self.verbose = verbose self.verbose = verbose
return self.parse_response(h.getfile()) return self.parse_response(h.getfile())
def make_connection(self, host): def make_connection(self, host):
# create a HTTP connection object from a host descriptor # create a HTTP connection object from a host descriptor
import httplib import httplib
return httplib.HTTP(host) return httplib.HTTP(host)
def send_request(self, connection, handler, request_body): def send_request(self, connection, handler, request_body):
connection.putrequest("POST", handler) connection.putrequest("POST", handler)
def send_host(self, connection, host): def send_host(self, connection, host):
connection.putheader("Host", host) connection.putheader("Host", host)
def send_user_agent(self, connection): def send_user_agent(self, connection):
connection.putheader("User-Agent", self.user_agent) connection.putheader("User-Agent", self.user_agent)
def send_content(self, connection, request_body): def send_content(self, connection, request_body):
connection.putheader("Content-Type", "text/xml") connection.putheader("Content-Type", "text/xml")
connection.putheader("Content-Length", str(len(request_body))) connection.putheader("Content-Length", str(len(request_body)))
connection.endheaders() connection.endheaders()
if request_body: if request_body:
connection.send(request_body) connection.send(request_body)
def parse_response(self, f): def parse_response(self, f):
# read response from input file, and parse it # read response from input file, and parse it
p, u = getparser() p, u = getparser()
while 1: while 1:
response = f.read(1024) response = f.read(1024)
if not response: if not response:
break break
if self.verbose: if self.verbose:
print "body:", repr(response) print "body:", repr(response)
p.feed(response) p.feed(response)
f.close() f.close()
p.close() p.close()
return u.close() return u.close()
class SafeTransport(Transport): class SafeTransport(Transport):
"""Handles an HTTPS transaction to an XML-RPC server""" """Handles an HTTPS transaction to an XML-RPC server"""
def make_connection(self, host): def make_connection(self, host):
# create a HTTPS connection object from a host descriptor # create a HTTPS connection object from a host descriptor
# host may be a string, or a (host, x509-dict) tuple # host may be a string, or a (host, x509-dict) tuple
import httplib import httplib
if isinstance(host, TupleType): if isinstance(host, TupleType):
host, x509 = host host, x509 = host
else: else:
x509 = {} x509 = {}
try: try:
HTTPS = httplib.HTTPS HTTPS = httplib.HTTPS
except AttributeError: except AttributeError:
raise NotImplementedError,\ raise NotImplementedError,\
"your version of httplib doesn't support HTTPS" "your version of httplib doesn't support HTTPS"
else: else:
return apply(HTTPS, (host, None), x509) return apply(HTTPS, (host, None), x509)
def send_host(self, connection, host): def send_host(self, connection, host):
if isinstance(host, TupleType): if isinstance(host, TupleType):
host, x509 = host host, x509 = host
connection.putheader("Host", host) connection.putheader("Host", host)
class Server: class Server:
"""uri [,options] -> a logical connection to an XML-RPC server """uri [,options] -> a logical connection to an XML-RPC server
uri is the connection point on the server, given as uri is the connection point on the server, given as
scheme://host/target. scheme://host/target.
The standard implementation always supports the "http" scheme. If The standard implementation always supports the "http" scheme. If
SSL socket support is available (Python 2.0), it also supports SSL socket support is available (Python 2.0), it also supports
"https". "https".
If the target part and the slash preceding it are both omitted, If the target part and the slash preceding it are both omitted,
"/RPC2" is assumed. "/RPC2" is assumed.
The following options can be given as keyword arguments: The following options can be given as keyword arguments:
transport: a transport factory transport: a transport factory
encoding: the request encoding (default is UTF-8) encoding: the request encoding (default is UTF-8)
All 8-bit strings passed to the server proxy are assumed to use All 8-bit strings passed to the server proxy are assumed to use
the given encoding. the given encoding.
""" """
def __init__(self, uri, transport=None, encoding=None, verbose=0): def __init__(self, uri, transport=None, encoding=None, verbose=0):
# establish a "logical" server connection # establish a "logical" server connection
# get the url # get the url
type, uri = urllib.splittype(uri) type, uri = urllib.splittype(uri)
if type not in ("http", "https"): if type not in ("http", "https"):
raise IOError, "unsupported XML-RPC protocol" raise IOError, "unsupported XML-RPC protocol"
self.__host, self.__handler = urllib.splithost(uri) self.__host, self.__handler = urllib.splithost(uri)
if not self.__handler: if not self.__handler:
self.__handler = "/RPC2" self.__handler = "/RPC2"
if transport is None: if transport is None:
if type == "https": if type == "https":
transport = SafeTransport() transport = SafeTransport()
else: else:
transport = Transport() transport = Transport()
self.__transport = transport self.__transport = transport
self.__encoding = encoding self.__encoding = encoding
self.__verbose = verbose self.__verbose = verbose
def __request(self, methodname, params): def __request(self, methodname, params):
# call a method on the remote server # call a method on the remote server
request = dumps(params, methodname, encoding=self.__encoding) request = dumps(params, methodname, encoding=self.__encoding)
response = self.__transport.request( response = self.__transport.request(
self.__host, self.__host,
self.__handler, self.__handler,
request, request,
verbose=self.__verbose verbose=self.__verbose
) )
if len(response) == 1: if len(response) == 1:
response = response[0] response = response[0]
return response return response
def __repr__(self): def __repr__(self):
return ( return (
"<Server proxy for %s%s>" % "<Server proxy for %s%s>" %
(self.__host, self.__handler) (self.__host, self.__handler)
) )
__str__ = __repr__ __str__ = __repr__
def __getattr__(self, name): def __getattr__(self, name):
# magic method dispatcher # magic method dispatcher
return _Method(self.__request, name) return _Method(self.__request, name)
# note: to call a remote object with an non-standard name, use # note: to call a remote object with an non-standard name, use
# result getattr(server, "strange-python-name")(args) # result getattr(server, "strange-python-name")(args)
# -------------------------------------------------------------------- # --------------------------------------------------------------------
# test code # test code
if __name__ == "__main__": if __name__ == "__main__":
# simple test program (from the XML-RPC specification) # simple test program (from the XML-RPC specification)
# server = Server("http://localhost:8000") # local server # server = Server("http://localhost:8000") # local server
server = Server("http://betty.userland.com") server = Server("http://betty.userland.com")
print server print server
try: try:
print server.examples.getStateName(41) print server.examples.getStateName(41)
except Error, v: except Error, v:
print "ERROR", v print "ERROR", v
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment