Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
Z
Zope
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
Kirill Smelkov
Zope
Commits
58cbf3d3
Commit
58cbf3d3
authored
May 31, 2002
by
Toby Dickenson
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
fixed line endings
parent
2eede254
Changes
1
Hide whitespace changes
Inline
Side-by-side
Showing
1 changed file
with
854 additions
and
854 deletions
+854
-854
lib/python/xmlrpclib.py
lib/python/xmlrpclib.py
+854
-854
No files found.
lib/python/xmlrpclib.py
View file @
58cbf3d3
#
#
# 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
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment