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
26370f24
Commit
26370f24
authored
Sep 03, 2016
by
Hanno Schlichting
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Move xmlrpc code into ZServer distribution.
parent
478ae3a2
Changes
9
Hide whitespace changes
Inline
Side-by-side
Showing
9 changed files
with
62 additions
and
452 deletions
+62
-452
src/ZPublisher/BaseRequest.py
src/ZPublisher/BaseRequest.py
+11
-3
src/ZPublisher/HTTPRequest.py
src/ZPublisher/HTTPRequest.py
+14
-15
src/ZPublisher/HTTPResponse.py
src/ZPublisher/HTTPResponse.py
+1
-7
src/ZPublisher/Publish.py
src/ZPublisher/Publish.py
+1
-19
src/ZPublisher/__init__.py
src/ZPublisher/__init__.py
+20
-1
src/ZPublisher/tests/test_WSGIPublisher.py
src/ZPublisher/tests/test_WSGIPublisher.py
+1
-1
src/ZPublisher/tests/test_xmlrpc.py
src/ZPublisher/tests/test_xmlrpc.py
+0
-214
src/ZPublisher/xmlrpc.py
src/ZPublisher/xmlrpc.py
+11
-189
src/Zope2/App/startup.py
src/Zope2/App/startup.py
+3
-3
No files found.
src/ZPublisher/BaseRequest.py
View file @
26370f24
...
@@ -15,12 +15,12 @@
...
@@ -15,12 +15,12 @@
from
urllib
import
quote
as
urllib_quote
from
urllib
import
quote
as
urllib_quote
import
types
import
types
import
xmlrpc
from
AccessControl.ZopeSecurityPolicy
import
getRoles
from
AccessControl.ZopeSecurityPolicy
import
getRoles
from
Acquisition
import
aq_base
,
aq_inner
from
Acquisition
import
aq_base
,
aq_inner
from
Acquisition.interfaces
import
IAcquirer
from
Acquisition.interfaces
import
IAcquirer
from
ExtensionClass
import
Base
from
ExtensionClass
import
Base
import
pkg_resources
from
zExceptions
import
Forbidden
from
zExceptions
import
Forbidden
from
zExceptions
import
NotFound
from
zExceptions
import
NotFound
from
zope.component
import
queryMultiAdapter
from
zope.component
import
queryMultiAdapter
...
@@ -39,6 +39,14 @@ from zope.traversing.namespace import nsParse
...
@@ -39,6 +39,14 @@ from zope.traversing.namespace import nsParse
from
ZPublisher.Converters
import
type_converters
from
ZPublisher.Converters
import
type_converters
from
ZPublisher.interfaces
import
UseTraversalDefault
from
ZPublisher.interfaces
import
UseTraversalDefault
try
:
dist
=
pkg_resources
.
get_distribution
(
'ZServer'
)
except
pkg_resources
.
DistributionNotFound
:
def
is_xmlrpc_response
(
response
):
return
False
else
:
from
ZServer.ZPublisher.xmlrpc
import
is_xmlrpc_response
_marker
=
[]
_marker
=
[]
UNSPECIFIED_ROLES
=
''
UNSPECIFIED_ROLES
=
''
...
@@ -386,8 +394,8 @@ class BaseRequest:
...
@@ -386,8 +394,8 @@ class BaseRequest:
# How did this request come in? (HTTP GET, PUT, POST, etc.)
# How did this request come in? (HTTP GET, PUT, POST, etc.)
method
=
request_get
(
'REQUEST_METHOD'
,
'GET'
).
upper
()
method
=
request_get
(
'REQUEST_METHOD'
,
'GET'
).
upper
()
if
(
method
in
[
"GET"
,
"POST"
,
"PURGE"
]
and
if
(
method
in
(
'GET'
,
'POST'
,
'PURGE'
)
and
not
is
instance
(
response
,
xmlrpc
.
R
esponse
)):
not
is
_xmlrpc_response
(
r
esponse
)):
# Probably a browser
# Probably a browser
no_acquire_flag
=
0
no_acquire_flag
=
0
# index_html is still the default method, only any object can
# index_html is still the default method, only any object can
...
...
src/ZPublisher/HTTPRequest.py
View file @
26370f24
...
@@ -14,6 +14,7 @@
...
@@ -14,6 +14,7 @@
""" HTTP request management.
""" HTTP request management.
"""
"""
import
base64
from
cgi
import
escape
from
cgi
import
escape
from
cgi
import
FieldStorage
from
cgi
import
FieldStorage
import
codecs
import
codecs
...
@@ -33,6 +34,8 @@ from urllib import unquote
...
@@ -33,6 +34,8 @@ from urllib import unquote
from
urllib
import
splittype
from
urllib
import
splittype
from
urllib
import
splitport
from
urllib
import
splitport
from
AccessControl.tainted
import
TaintedString
import
pkg_resources
from
zope.i18n.interfaces
import
IUserPreferredLanguages
from
zope.i18n.interfaces
import
IUserPreferredLanguages
from
zope.i18n.locales
import
locales
,
LoadLocaleError
from
zope.i18n.locales
import
locales
,
LoadLocaleError
from
zope.interface
import
directlyProvidedBy
from
zope.interface
import
directlyProvidedBy
...
@@ -41,7 +44,6 @@ from zope.interface import implements
...
@@ -41,7 +44,6 @@ from zope.interface import implements
from
zope.publisher.base
import
DebugFlags
from
zope.publisher.base
import
DebugFlags
from
zope.publisher.interfaces.browser
import
IBrowserRequest
from
zope.publisher.interfaces.browser
import
IBrowserRequest
from
AccessControl.tainted
import
TaintedString
from
ZPublisher.BaseRequest
import
BaseRequest
from
ZPublisher.BaseRequest
import
BaseRequest
from
ZPublisher.BaseRequest
import
quote
from
ZPublisher.BaseRequest
import
quote
from
ZPublisher.Converters
import
get_converter
from
ZPublisher.Converters
import
get_converter
...
@@ -49,6 +51,14 @@ from ZPublisher.Converters import get_converter
...
@@ -49,6 +51,14 @@ from ZPublisher.Converters import get_converter
if
sys
.
version_info
>=
(
3
,
):
if
sys
.
version_info
>=
(
3
,
):
unicode
=
str
unicode
=
str
xmlrpc
=
None
try
:
dist
=
pkg_resources
.
get_distribution
(
'ZServer'
)
except
pkg_resources
.
DistributionNotFound
:
pass
else
:
from
ZServer.ZPublisher
import
xmlrpc
# Flags
# Flags
SEQUENCE
=
1
SEQUENCE
=
1
DEFAULT
=
2
DEFAULT
=
2
...
@@ -58,10 +68,6 @@ REC = RECORD | RECORDS
...
@@ -58,10 +68,6 @@ REC = RECORD | RECORDS
EMPTY
=
16
EMPTY
=
16
CONVERTED
=
32
CONVERTED
=
32
# Placeholders for module that we'll import if we have to.
xmlrpc
=
None
base64
=
None
# This may get overwritten during configuration
# This may get overwritten during configuration
default_encoding
=
'utf-8'
default_encoding
=
'utf-8'
...
@@ -504,14 +510,10 @@ class HTTPRequest(BaseRequest):
...
@@ -504,14 +510,10 @@ class HTTPRequest(BaseRequest):
if
'HTTP_SOAPACTION'
in
environ
:
if
'HTTP_SOAPACTION'
in
environ
:
# Stash XML request for interpretation by a SOAP-aware view
# Stash XML request for interpretation by a SOAP-aware view
other
[
'SOAPXML'
]
=
fs
.
value
other
[
'SOAPXML'
]
=
fs
.
value
# Hm, maybe it's an XML-RPC
elif
(
xmlrpc
is
not
None
and
method
==
'POST'
and
elif
(
'content-type'
in
fs
.
headers
and
(
'content-type'
in
fs
.
headers
and
'text/xml'
in
fs
.
headers
[
'content-type'
]
and
'text/xml'
in
fs
.
headers
[
'content-type'
])):
method
==
'POST'
):
# Ye haaa, XML-RPC!
# Ye haaa, XML-RPC!
global
xmlrpc
if
xmlrpc
is
None
:
from
ZPublisher
import
xmlrpc
meth
,
self
.
args
=
xmlrpc
.
parse_input
(
fs
.
value
)
meth
,
self
.
args
=
xmlrpc
.
parse_input
(
fs
.
value
)
response
=
xmlrpc
.
response
(
response
)
response
=
xmlrpc
.
response
(
response
)
other
[
'RESPONSE'
]
=
self
.
response
=
response
other
[
'RESPONSE'
]
=
self
.
response
=
response
...
@@ -1528,12 +1530,9 @@ class HTTPRequest(BaseRequest):
...
@@ -1528,12 +1530,9 @@ class HTTPRequest(BaseRequest):
return
result
return
result
def
_authUserPW
(
self
):
def
_authUserPW
(
self
):
global
base64
auth
=
self
.
_auth
auth
=
self
.
_auth
if
auth
:
if
auth
:
if
auth
[:
6
].
lower
()
==
'basic '
:
if
auth
[:
6
].
lower
()
==
'basic '
:
if
base64
is
None
:
import
base64
[
name
,
password
]
=
\
[
name
,
password
]
=
\
base64
.
decodestring
(
auth
.
split
()[
-
1
]).
split
(
':'
,
1
)
base64
.
decodestring
(
auth
.
split
()[
-
1
]).
split
(
':'
,
1
)
return
name
,
password
return
name
,
password
...
...
src/ZPublisher/HTTPResponse.py
View file @
26370f24
...
@@ -517,8 +517,7 @@ class HTTPResponse(BaseResponse):
...
@@ -517,8 +517,7 @@ class HTTPResponse(BaseResponse):
This function is designed to be called on each request to specify
This function is designed to be called on each request to specify
on a request-by-request basis that the response content should
on a request-by-request basis that the response content should
be compressed. This is quite useful for xml-rpc transactions, where
be compressed.
compression rates of 90% or more can be achieved for text data.
The REQUEST headers are used to determine if the client accepts
The REQUEST headers are used to determine if the client accepts
gzip content encoding. The force parameter can force the use
gzip content encoding. The force parameter can force the use
...
@@ -705,11 +704,6 @@ class HTTPResponse(BaseResponse):
...
@@ -705,11 +704,6 @@ class HTTPResponse(BaseResponse):
et
=
translate
(
str
(
t
),
nl2sp
)
et
=
translate
(
str
(
t
),
nl2sp
)
self
.
setHeader
(
'bobo-exception-type'
,
et
)
self
.
setHeader
(
'bobo-exception-type'
,
et
)
# As of Zope 2.6.2 / 2.7, we no longer try to pass along a
# meaningful exception value. Now that there are good logging
# facilities on the server side (and given that xml-rpc has
# largely removed the need for this code at all), we just
# refer the caller to the server error log.
ev
=
'See the server error log for details'
ev
=
'See the server error log for details'
self
.
setHeader
(
'bobo-exception-value'
,
ev
)
self
.
setHeader
(
'bobo-exception-value'
,
ev
)
...
...
src/ZPublisher/Publish.py
View file @
26370f24
...
@@ -32,29 +32,11 @@ from zope.security.management import newInteraction, endInteraction
...
@@ -32,29 +32,11 @@ from zope.security.management import newInteraction, endInteraction
from
ZPublisher.mapply
import
mapply
from
ZPublisher.mapply
import
mapply
from
ZPublisher
import
pubevents
from
ZPublisher
import
pubevents
from
ZPublisher
import
Retry
from
ZPublisher.HTTPRequest
import
HTTPRequest
as
Request
from
ZPublisher.HTTPRequest
import
HTTPRequest
as
Request
from
ZPublisher.HTTPResponse
import
HTTPResponse
as
Response
from
ZPublisher.HTTPResponse
import
HTTPResponse
as
Response
class
Retry
(
Exception
):
"""Raise this to retry a request
"""
def
__init__
(
self
,
t
=
None
,
v
=
None
,
tb
=
None
):
self
.
_args
=
t
,
v
,
tb
def
reraise
(
self
):
t
,
v
,
tb
=
self
.
_args
if
t
is
None
:
t
=
Retry
if
tb
is
None
:
raise
t
(
v
)
try
:
reraise
(
t
,
v
,
tb
)
finally
:
tb
=
None
def
call_object
(
object
,
args
,
request
):
def
call_object
(
object
,
args
,
request
):
return
object
(
*
args
)
return
object
(
*
args
)
...
...
src/ZPublisher/__init__.py
View file @
26370f24
...
@@ -11,6 +11,25 @@
...
@@ -11,6 +11,25 @@
#
#
##############################################################################
##############################################################################
from
six
import
reraise
from
zExceptions
import
NotFound
,
BadRequest
,
InternalError
,
Forbidden
# NOQA
from
zExceptions
import
NotFound
,
BadRequest
,
InternalError
,
Forbidden
# NOQA
from
ZPublisher.Publish
import
publish_module
,
Retry
# NOQA
class
Retry
(
Exception
):
"""Raise this to retry a request
"""
def
__init__
(
self
,
t
=
None
,
v
=
None
,
tb
=
None
):
self
.
_args
=
t
,
v
,
tb
def
reraise
(
self
):
t
,
v
,
tb
=
self
.
_args
if
t
is
None
:
t
=
Retry
if
tb
is
None
:
raise
t
(
v
)
try
:
reraise
(
t
,
v
,
tb
)
finally
:
tb
=
None
src/ZPublisher/tests/test_WSGIPublisher.py
View file @
26370f24
...
@@ -12,7 +12,7 @@
...
@@ -12,7 +12,7 @@
##############################################################################
##############################################################################
import
unittest
import
unittest
from
ZPublisher.
Publish
import
get_module_info
from
ZPublisher.
WSGIPublisher
import
get_module_info
class
WSGIResponseTests
(
unittest
.
TestCase
):
class
WSGIResponseTests
(
unittest
.
TestCase
):
...
...
src/ZPublisher/tests/test_xmlrpc.py
deleted
100644 → 0
View file @
478ae3a2
import
unittest
from
DateTime
import
DateTime
class
FauxResponse
:
def
__init__
(
self
):
self
.
_headers
=
{}
self
.
_body
=
None
def
setBody
(
self
,
body
):
self
.
_body
=
body
def
setHeader
(
self
,
name
,
value
):
self
.
_headers
[
name
]
=
value
def
setStatus
(
self
,
status
):
self
.
_status
=
status
class
FauxInstance
:
def
__init__
(
self
,
**
kw
):
self
.
__dict__
.
update
(
kw
)
class
XMLRPCResponseTests
(
unittest
.
TestCase
):
def
_getTargetClass
(
self
):
from
ZPublisher.xmlrpc
import
Response
return
Response
def
_makeOne
(
self
,
*
args
,
**
kw
):
return
self
.
_getTargetClass
()(
*
args
,
**
kw
)
def
test_setBody
(
self
):
import
xmlrpclib
body
=
FauxInstance
(
_secret
=
'abc'
,
public
=
'def'
)
faux
=
FauxResponse
()
response
=
self
.
_makeOne
(
faux
)
response
.
setBody
(
body
)
body_str
=
faux
.
_body
self
.
assertEqual
(
type
(
body_str
),
type
(
''
))
as_set
,
method
=
xmlrpclib
.
loads
(
body_str
)
as_set
=
as_set
[
0
]
self
.
assertEqual
(
method
,
None
)
self
.
assertFalse
(
'_secret'
in
as_set
.
keys
())
self
.
assertTrue
(
'public'
in
as_set
.
keys
())
self
.
assertEqual
(
as_set
[
'public'
],
'def'
)
def
test_nil
(
self
):
import
xmlrpclib
body
=
FauxInstance
(
public
=
None
)
faux
=
FauxResponse
()
response
=
self
.
_makeOne
(
faux
)
response
.
setBody
(
body
)
data
,
method
=
xmlrpclib
.
loads
(
faux
.
_body
)
self
.
assert_
(
data
[
0
][
'public'
]
is
None
)
def
test_instance
(
self
):
# Instances are turned into dicts with their private
# attributes removed.
import
xmlrpclib
body
=
FauxInstance
(
_secret
=
'abc'
,
public
=
'def'
)
faux
=
FauxResponse
()
response
=
self
.
_makeOne
(
faux
)
response
.
setBody
(
body
)
data
,
method
=
xmlrpclib
.
loads
(
faux
.
_body
)
data
=
data
[
0
]
self
.
assertEqual
(
data
,
{
'public'
:
'def'
})
def
test_instanceattribute
(
self
):
# While the removal of private ('_') attributes works fine for the
# top-level instance, how about attributes that are themselves
# instances?
import
xmlrpclib
body
=
FauxInstance
(
public
=
FauxInstance
(
_secret
=
'abc'
,
public
=
'def'
))
faux
=
FauxResponse
()
response
=
self
.
_makeOne
(
faux
)
response
.
setBody
(
body
)
data
,
method
=
xmlrpclib
.
loads
(
faux
.
_body
)
data
=
data
[
0
][
'public'
]
self
.
assertEqual
(
data
,
{
'public'
:
'def'
})
def
test_instanceattribute_recursive
(
self
):
# Instance "flattening" should work recursively, ad infinitum
import
xmlrpclib
body
=
FauxInstance
(
public
=
FauxInstance
(
public
=
FauxInstance
(
_secret
=
'abc'
,
public
=
'def'
)))
faux
=
FauxResponse
()
response
=
self
.
_makeOne
(
faux
)
response
.
setBody
(
body
)
data
,
method
=
xmlrpclib
.
loads
(
faux
.
_body
)
data
=
data
[
0
][
'public'
][
'public'
]
self
.
assertEqual
(
data
,
{
'public'
:
'def'
})
def
test_instance_in_list
(
self
):
# Instances are turned into dicts with their private
# attributes removed, even when embedded in another
# data structure.
import
xmlrpclib
body
=
[
FauxInstance
(
_secret
=
'abc'
,
public
=
'def'
)]
faux
=
FauxResponse
()
response
=
self
.
_makeOne
(
faux
)
response
.
setBody
(
body
)
data
,
method
=
xmlrpclib
.
loads
(
faux
.
_body
)
data
=
data
[
0
][
0
]
self
.
assertEqual
(
data
,
{
'public'
:
'def'
})
def
test_instance_in_dict
(
self
):
# Instances are turned into dicts with their private
# attributes removed, even when embedded in another
# data structure.
import
xmlrpclib
body
=
{
'faux'
:
FauxInstance
(
_secret
=
'abc'
,
public
=
'def'
)}
faux
=
FauxResponse
()
response
=
self
.
_makeOne
(
faux
)
response
.
setBody
(
body
)
data
,
method
=
xmlrpclib
.
loads
(
faux
.
_body
)
data
=
data
[
0
][
'faux'
]
self
.
assertEqual
(
data
,
{
'public'
:
'def'
})
def
test_zopedatetimeinstance
(
self
):
# DateTime instance at top-level
import
xmlrpclib
body
=
DateTime
(
'2006-05-24 07:00:00 GMT+0'
)
faux
=
FauxResponse
()
response
=
self
.
_makeOne
(
faux
)
response
.
setBody
(
body
)
data
,
method
=
xmlrpclib
.
loads
(
faux
.
_body
)
data
=
data
[
0
]
self
.
assertTrue
(
isinstance
(
data
,
xmlrpclib
.
DateTime
))
self
.
assertEqual
(
data
.
value
,
u'2006-05-24T07:00:00+00:00'
)
def
test_zopedatetimeattribute
(
self
):
# DateTime instance as attribute
import
xmlrpclib
body
=
FauxInstance
(
public
=
DateTime
(
'2006-05-24 07:00:00 GMT+0'
))
faux
=
FauxResponse
()
response
=
self
.
_makeOne
(
faux
)
response
.
setBody
(
body
)
data
,
method
=
xmlrpclib
.
loads
(
faux
.
_body
)
data
=
data
[
0
][
'public'
]
self
.
assertTrue
(
isinstance
(
data
,
xmlrpclib
.
DateTime
))
self
.
assertEqual
(
data
.
value
,
u'2006-05-24T07:00:00+00:00'
)
def
test_zopedatetimeattribute_recursive
(
self
):
# DateTime encoding should work recursively
import
xmlrpclib
body
=
FauxInstance
(
public
=
FauxInstance
(
public
=
DateTime
(
'2006-05-24 07:00:00 GMT+0'
)))
faux
=
FauxResponse
()
response
=
self
.
_makeOne
(
faux
)
response
.
setBody
(
body
)
data
,
method
=
xmlrpclib
.
loads
(
faux
.
_body
)
data
=
data
[
0
][
'public'
][
'public'
]
self
.
assertTrue
(
isinstance
(
data
,
xmlrpclib
.
DateTime
))
self
.
assertEqual
(
data
.
value
,
u'2006-05-24T07:00:00+00:00'
)
def
test_zopedatetimeinstance_in_list
(
self
):
# DateTime instance embedded in a list
import
xmlrpclib
body
=
[
DateTime
(
'2006-05-24 07:00:00 GMT+0'
)]
faux
=
FauxResponse
()
response
=
self
.
_makeOne
(
faux
)
response
.
setBody
(
body
)
data
,
method
=
xmlrpclib
.
loads
(
faux
.
_body
)
data
=
data
[
0
][
0
]
self
.
assertTrue
(
isinstance
(
data
,
xmlrpclib
.
DateTime
))
self
.
assertEqual
(
data
.
value
,
u'2006-05-24T07:00:00+00:00'
)
def
test_zopedatetimeinstance_in_dict
(
self
):
# DateTime instance embedded in a dict
import
xmlrpclib
body
=
{
'date'
:
DateTime
(
'2006-05-24 07:00:00 GMT+0'
)}
faux
=
FauxResponse
()
response
=
self
.
_makeOne
(
faux
)
response
.
setBody
(
body
)
data
,
method
=
xmlrpclib
.
loads
(
faux
.
_body
)
data
=
data
[
0
][
'date'
]
self
.
assertTrue
(
isinstance
(
data
,
xmlrpclib
.
DateTime
))
self
.
assertEqual
(
data
.
value
,
u'2006-05-24T07:00:00+00:00'
)
def
test_functionattribute
(
self
):
# Cannot marshal functions or methods, obviously
import
xmlrpclib
def
foo
():
pass
body
=
FauxInstance
(
public
=
foo
)
faux
=
FauxResponse
()
response
=
self
.
_makeOne
(
faux
)
response
.
setBody
(
body
)
func
=
xmlrpclib
.
loads
(
faux
.
_body
)
self
.
assertEqual
(
func
,
(({
'public'
:
{}},),
None
))
def
test_emptystringattribute
(
self
):
# Test an edge case: attribute name '' is possible,
# at least in theory.
import
xmlrpclib
body
=
FauxInstance
(
_secret
=
'abc'
)
setattr
(
body
,
''
,
True
)
faux
=
FauxResponse
()
response
=
self
.
_makeOne
(
faux
)
response
.
setBody
(
body
)
data
,
method
=
xmlrpclib
.
loads
(
faux
.
_body
)
data
=
data
[
0
]
self
.
assertEqual
(
data
,
{
''
:
True
})
src/ZPublisher/xmlrpc.py
View file @
26370f24
...
@@ -10,193 +10,15 @@
...
@@ -10,193 +10,15 @@
# FOR A PARTICULAR PURPOSE.
# FOR A PARTICULAR PURPOSE.
#
#
##############################################################################
##############################################################################
"""XML-RPC support module
Written by Eric Kidd at UserLand software, with much help from Jim Fulton
from
zope.deferredimport
import
deprecated
at DC. This code hooks Zope up to Fredrik Lundh's Python XML-RPC library.
# BBB Zope 5.0
See http://www.xmlrpc.com/ and http://linux.userland.com/ for more
deprecated
(
information about XML-RPC and Zope.
'Please import from ZServer.ZPublisher.xmlrpc.'
,
"""
dump_instance
=
'ZServer.ZPublisher.xmlrpc:dump_instance'
,
parse_input
=
'ZServer.ZPublisher.xmlrpc:parse_input'
,
import
re
response
=
'ZServer.ZPublisher.xmlrpc:response'
,
import
sys
Response
=
'ZServer.ZPublisher.xmlrpc:Response'
,
import
types
WRAPPERS
=
'ZServer.ZPublisher.xmlrpc:WRAPPERS'
,
import
xmlrpclib
)
from
App.config
import
getConfiguration
from
zExceptions
import
Unauthorized
from
ZODB.POSException
import
ConflictError
# Make DateTime.DateTime marshallable via XML-RPC
# http://www.zope.org/Collectors/Zope/2109
from
DateTime.DateTime
import
DateTime
WRAPPERS
=
xmlrpclib
.
WRAPPERS
+
(
DateTime
,
)
def
dump_instance
(
self
,
value
,
write
):
# Check for special wrappers
if
value
.
__class__
in
WRAPPERS
:
self
.
write
=
write
value
.
encode
(
self
)
del
self
.
write
else
:
# Store instance attributes as a struct (really?).
# We want to avoid disclosing private attributes.
# Private attributes are by convention named with
# a leading underscore character.
value
=
dict
([(
k
,
v
)
for
(
k
,
v
)
in
value
.
__dict__
.
items
()
if
k
[:
1
]
!=
'_'
])
self
.
dump_struct
(
value
,
write
)
xmlrpclib
.
Marshaller
.
dispatch
[
types
.
InstanceType
]
=
dump_instance
xmlrpclib
.
Marshaller
.
dispatch
[
DateTime
]
=
dump_instance
def
parse_input
(
data
):
"""Parse input data and return a method path and argument tuple
The data is a string.
"""
#
# For example, with the input:
#
# <?xml version="1.0"?>
# <methodCall>
# <methodName>examples.getStateName</methodName>
# <params>
# <param>
# <value><i4>41</i4></value>
# </param>
# </params>
# </methodCall>
#
# the function should return:
#
# ('examples.getStateName', (41,))
params
,
method
=
xmlrpclib
.
loads
(
data
)
# Translate '.' to '/' in meth to represent object traversal.
method
=
method
.
replace
(
'.'
,
'/'
)
return
method
,
params
# See below
#
# def response(anHTTPResponse):
# """Return a valid ZPublisher response object
#
# Use data already gathered by the existing response.
# The new response will replace the existing response.
# """
# # As a first cut, lets just clone the response and
# # put all of the logic in our refined response class below.
# r=Response()
# r.__dict__.update(anHTTPResponse.__dict__)
# return r
########################################################################
# Possible implementation helpers:
class
Response
:
"""Customized Response that handles XML-RPC-specific details.
We override setBody to marhsall Python objects into XML-RPC. We
also override exception to convert errors to XML-RPC faults.
If these methods stop getting called, make sure that ZPublisher is
using the xmlrpc.Response object created above and not the original
HTTPResponse object from which it was cloned.
It's probably possible to improve the 'exception' method quite a bit.
The current implementation, however, should suffice for now.
"""
_error_format
=
'text/plain'
# No html in error values
# Because we can't predict what kind of thing we're customizing,
# we have to use delegation, rather than inheritance to do the
# customization.
def
__init__
(
self
,
real
):
self
.
__dict__
[
'_real'
]
=
real
def
__getattr__
(
self
,
name
):
return
getattr
(
self
.
_real
,
name
)
def
__setattr__
(
self
,
name
,
v
):
return
setattr
(
self
.
_real
,
name
,
v
)
def
__delattr__
(
self
,
name
):
return
delattr
(
self
.
_real
,
name
)
def
setBody
(
self
,
body
,
title
=
''
,
is_error
=
0
,
bogus_str_search
=
None
):
if
isinstance
(
body
,
xmlrpclib
.
Fault
):
# Convert Fault object to XML-RPC response.
body
=
xmlrpclib
.
dumps
(
body
,
methodresponse
=
1
,
allow_none
=
True
)
else
:
# Marshall our body as an XML-RPC response. Strings will be sent
# strings, integers as integers, etc. We do *not* convert
# everything to a string first.
# Previously this had special handling if the response
# was a Python None. This is now patched in xmlrpclib to
# allow Nones nested inside data structures too.
try
:
body
=
xmlrpclib
.
dumps
(
(
body
,),
methodresponse
=
1
,
allow_none
=
True
)
except
ConflictError
:
raise
except
:
self
.
exception
()
return
# Set our body to the XML-RPC message, and fix our MIME type.
self
.
_real
.
setBody
(
body
)
self
.
_real
.
setHeader
(
'content-type'
,
'text/xml'
)
return
self
def
exception
(
self
,
fatal
=
0
,
info
=
None
,
absuri_match
=
None
,
tag_search
=
None
):
# Fetch our exception info. t is type, v is value and tb is the
# traceback object.
if
isinstance
(
info
,
tuple
)
and
len
(
info
)
==
3
:
t
,
v
,
tb
=
info
else
:
t
,
v
,
tb
=
sys
.
exc_info
()
# Don't mask 404 respnses, as some XML-RPC libraries rely on the HTTP
# mechanisms for detecting when authentication is required. Fixes Zope
# Collector issue 525.
if
issubclass
(
t
,
Unauthorized
):
return
self
.
_real
.
exception
(
fatal
=
fatal
,
info
=
info
)
# Create an appropriate Fault object. Containing error information
Fault
=
xmlrpclib
.
Fault
f
=
None
try
:
# Strip HTML tags from the error value
vstr
=
str
(
v
)
remove
=
[
r"<[^<>]*>"
,
r"&[A-Za-z]+;"
]
for
pat
in
remove
:
vstr
=
re
.
sub
(
pat
,
" "
,
vstr
)
if
getConfiguration
().
debug_mode
:
from
traceback
import
format_exception
value
=
'
\
n
'
+
''
.
join
(
format_exception
(
t
,
vstr
,
tb
))
else
:
value
=
'%s - %s'
%
(
t
,
vstr
)
if
isinstance
(
v
,
Fault
):
f
=
v
elif
isinstance
(
v
,
Exception
):
f
=
Fault
(
-
1
,
'Unexpected Zope exception: %s'
%
value
)
else
:
f
=
Fault
(
-
2
,
'Unexpected Zope error value: %s'
%
value
)
except
ConflictError
:
raise
except
Exception
:
f
=
Fault
(
-
3
,
"Unknown Zope fault type"
)
# Do the damage.
self
.
setBody
(
f
)
self
.
_real
.
setStatus
(
200
)
return
tb
response
=
Response
src/Zope2/App/startup.py
View file @
26370f24
...
@@ -27,7 +27,7 @@ import OFS.Application
...
@@ -27,7 +27,7 @@ import OFS.Application
import
transaction
import
transaction
import
ZODB
import
ZODB
import
Zope2
import
Zope2
import
ZPublisher
from
ZPublisher
import
Retry
from
AccessControl.SecurityManagement
import
newSecurityManager
from
AccessControl.SecurityManagement
import
newSecurityManager
from
AccessControl.SecurityManagement
import
noSecurityManager
from
AccessControl.SecurityManagement
import
noSecurityManager
from
Acquisition
import
aq_acquire
from
Acquisition
import
aq_acquire
...
@@ -200,9 +200,9 @@ class ZPublisherExceptionHook:
...
@@ -200,9 +200,9 @@ class ZPublisherExceptionHook:
if
issubclass
(
t
,
ConflictError
):
if
issubclass
(
t
,
ConflictError
):
self
.
logConflicts
(
v
,
REQUEST
)
self
.
logConflicts
(
v
,
REQUEST
)
raise
ZPublisher
.
Retry
(
t
,
v
,
traceback
)
raise
Retry
(
t
,
v
,
traceback
)
if
t
is
ZPublisher
.
Retry
:
if
t
is
Retry
:
try
:
try
:
v
.
reraise
()
v
.
reraise
()
except
:
except
:
...
...
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