Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
Z
ZODB
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
Nicolas Wavrant
ZODB
Commits
6d107ff8
Commit
6d107ff8
authored
Apr 06, 2011
by
Jim Fulton
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Provide shorter code path for loads, which are most common operation.
Simplified and optimized marshalling code.
parent
dc07ea83
Changes
3
Hide whitespace changes
Inline
Side-by-side
Showing
3 changed files
with
105 additions
and
61 deletions
+105
-61
src/ZEO/tests/ConnectionTests.py
src/ZEO/tests/ConnectionTests.py
+2
-2
src/ZEO/zrpc/connection.py
src/ZEO/zrpc/connection.py
+50
-11
src/ZEO/zrpc/marshal.py
src/ZEO/zrpc/marshal.py
+53
-48
No files found.
src/ZEO/tests/ConnectionTests.py
View file @
6d107ff8
...
@@ -24,7 +24,7 @@ import logging
...
@@ -24,7 +24,7 @@ import logging
import
ZEO.ServerStub
import
ZEO.ServerStub
from
ZEO.ClientStorage
import
ClientStorage
from
ZEO.ClientStorage
import
ClientStorage
from
ZEO.Exceptions
import
ClientDisconnected
from
ZEO.Exceptions
import
ClientDisconnected
from
ZEO.zrpc.marshal
import
Marshaller
from
ZEO.zrpc.marshal
import
encode
from
ZEO.tests
import
forker
from
ZEO.tests
import
forker
from
ZODB.DB
import
DB
from
ZODB.DB
import
DB
...
@@ -475,7 +475,7 @@ class ConnectionTests(CommonSetupTearDown):
...
@@ -475,7 +475,7 @@ class ConnectionTests(CommonSetupTearDown):
class
Hack
:
class
Hack
:
pass
pass
msg
=
Marshaller
().
encode
(
1
,
0
,
"foo"
,
(
Hack
(),))
msg
=
encode
(
1
,
0
,
"foo"
,
(
Hack
(),))
self
.
_bad_message
(
msg
)
self
.
_bad_message
(
msg
)
del
Hack
del
Hack
...
...
src/ZEO/zrpc/connection.py
View file @
6d107ff8
...
@@ -13,11 +13,13 @@
...
@@ -13,11 +13,13 @@
##############################################################################
##############################################################################
import
asyncore
import
asyncore
import
atexit
import
atexit
import
cPickle
import
errno
import
errno
import
select
import
select
import
sys
import
sys
import
threading
import
threading
import
logging
import
logging
import
ZEO.zrpc.marshal
import
traceback
,
time
import
traceback
,
time
...
@@ -25,7 +27,6 @@ import ZEO.zrpc.trigger
...
@@ -25,7 +27,6 @@ import ZEO.zrpc.trigger
from
ZEO.zrpc
import
smac
from
ZEO.zrpc
import
smac
from
ZEO.zrpc.error
import
ZRPCError
,
DisconnectedError
from
ZEO.zrpc.error
import
ZRPCError
,
DisconnectedError
from
ZEO.zrpc.marshal
import
Marshaller
,
ServerMarshaller
from
ZEO.zrpc.log
import
short_repr
,
log
from
ZEO.zrpc.log
import
short_repr
,
log
from
ZODB.loglevels
import
BLATHER
,
TRACE
from
ZODB.loglevels
import
BLATHER
,
TRACE
import
ZODB.POSException
import
ZODB.POSException
...
@@ -287,7 +288,10 @@ class Connection(smac.SizedMessageAsyncConnection, object):
...
@@ -287,7 +288,10 @@ class Connection(smac.SizedMessageAsyncConnection, object):
# our peer.
# our peer.
def
__init__
(
self
,
sock
,
addr
,
obj
,
tag
,
map
=
None
):
def
__init__
(
self
,
sock
,
addr
,
obj
,
tag
,
map
=
None
):
self
.
obj
=
None
self
.
obj
=
None
self
.
marshal
=
Marshaller
()
self
.
decode
=
ZEO
.
zrpc
.
marshal
.
decode
self
.
encode
=
ZEO
.
zrpc
.
marshal
.
encode
self
.
fast_encode
=
ZEO
.
zrpc
.
marshal
.
fast_encode
self
.
closed
=
False
self
.
closed
=
False
self
.
peer_protocol_version
=
None
# set in recv_handshake()
self
.
peer_protocol_version
=
None
# set in recv_handshake()
...
@@ -413,13 +417,34 @@ class Connection(smac.SizedMessageAsyncConnection, object):
...
@@ -413,13 +417,34 @@ class Connection(smac.SizedMessageAsyncConnection, object):
# will raise an exception. The exception will ultimately
# will raise an exception. The exception will ultimately
# result in asycnore calling handle_error(), which will
# result in asycnore calling handle_error(), which will
# close the connection.
# close the connection.
msgid
,
async
,
name
,
args
=
self
.
marshal
.
decode
(
message
)
msgid
,
async
,
name
,
args
=
self
.
decode
(
message
)
if
debug_zrpc
:
if
debug_zrpc
:
self
.
log
(
"recv msg: %s, %s, %s, %s"
%
(
msgid
,
async
,
name
,
self
.
log
(
"recv msg: %s, %s, %s, %s"
%
(
msgid
,
async
,
name
,
short_repr
(
args
)),
short_repr
(
args
)),
level
=
TRACE
)
level
=
TRACE
)
if
name
==
REPLY
:
if
name
==
'loadEx'
:
# Special case and inline the heck out of load case:
try
:
ret
=
self
.
obj
.
loadEx
(
*
args
)
except
(
SystemExit
,
KeyboardInterrupt
):
raise
except
Exception
,
msg
:
if
not
isinstance
(
msg
,
self
.
unlogged_exception_types
):
self
.
log
(
"%s() raised exception: %s"
%
(
name
,
msg
),
logging
.
ERROR
,
exc_info
=
True
)
self
.
return_error
(
msgid
,
*
sys
.
exc_info
()[:
2
])
else
:
try
:
self
.
message_output
(
self
.
fast_encode
(
msgid
,
0
,
REPLY
,
ret
))
self
.
poll
()
except
:
# Fall back to normal version for better error handling
self
.
send_reply
(
msgid
,
ret
)
elif
name
==
REPLY
:
assert
not
async
assert
not
async
self
.
handle_reply
(
msgid
,
args
)
self
.
handle_reply
(
msgid
,
args
)
else
:
else
:
...
@@ -493,14 +518,14 @@ class Connection(smac.SizedMessageAsyncConnection, object):
...
@@ -493,14 +518,14 @@ class Connection(smac.SizedMessageAsyncConnection, object):
# it's acceptable -- we really do want to catch every exception
# it's acceptable -- we really do want to catch every exception
# cPickle may raise.
# cPickle may raise.
try
:
try
:
msg
=
self
.
marshal
.
encode
(
msgid
,
0
,
REPLY
,
(
err_type
,
err_value
))
msg
=
self
.
encode
(
msgid
,
0
,
REPLY
,
(
err_type
,
err_value
))
except
:
# see above
except
:
# see above
try
:
try
:
r
=
short_repr
(
err_value
)
r
=
short_repr
(
err_value
)
except
:
except
:
r
=
"<unreprable>"
r
=
"<unreprable>"
err
=
ZRPCError
(
"Couldn't pickle error %.100s"
%
r
)
err
=
ZRPCError
(
"Couldn't pickle error %.100s"
%
r
)
msg
=
self
.
marshal
.
encode
(
msgid
,
0
,
REPLY
,
(
ZRPCError
,
err
))
msg
=
self
.
encode
(
msgid
,
0
,
REPLY
,
(
ZRPCError
,
err
))
self
.
message_output
(
msg
)
self
.
message_output
(
msg
)
self
.
poll
()
self
.
poll
()
...
@@ -527,7 +552,7 @@ class Connection(smac.SizedMessageAsyncConnection, object):
...
@@ -527,7 +552,7 @@ class Connection(smac.SizedMessageAsyncConnection, object):
if
debug_zrpc
:
if
debug_zrpc
:
self
.
log
(
"send msg: %d, %d, %s, ..."
%
(
msgid
,
async
,
method
),
self
.
log
(
"send msg: %d, %d, %s, ..."
%
(
msgid
,
async
,
method
),
level
=
TRACE
)
level
=
TRACE
)
buf
=
self
.
marshal
.
encode
(
msgid
,
async
,
method
,
args
)
buf
=
self
.
encode
(
msgid
,
async
,
method
,
args
)
self
.
message_output
(
buf
)
self
.
message_output
(
buf
)
return
msgid
return
msgid
...
@@ -560,7 +585,7 @@ class Connection(smac.SizedMessageAsyncConnection, object):
...
@@ -560,7 +585,7 @@ class Connection(smac.SizedMessageAsyncConnection, object):
The calls will not be interleaved with other calls from the same
The calls will not be interleaved with other calls from the same
client.
client.
"""
"""
self
.
message_output
(
self
.
marshal
.
encode
(
0
,
1
,
method
,
args
)
self
.
message_output
(
self
.
encode
(
0
,
1
,
method
,
args
)
for
method
,
args
in
iterator
)
for
method
,
args
in
iterator
)
def
handle_reply
(
self
,
msgid
,
ret
):
def
handle_reply
(
self
,
msgid
,
ret
):
...
@@ -573,6 +598,8 @@ class Connection(smac.SizedMessageAsyncConnection, object):
...
@@ -573,6 +598,8 @@ class Connection(smac.SizedMessageAsyncConnection, object):
self
.
trigger
.
pull_trigger
()
self
.
trigger
.
pull_trigger
()
# import cProfile, time
class
ManagedServerConnection
(
Connection
):
class
ManagedServerConnection
(
Connection
):
"""Server-side Connection subclass."""
"""Server-side Connection subclass."""
...
@@ -583,7 +610,9 @@ class ManagedServerConnection(Connection):
...
@@ -583,7 +610,9 @@ class ManagedServerConnection(Connection):
self
.
mgr
=
mgr
self
.
mgr
=
mgr
map
=
{}
map
=
{}
Connection
.
__init__
(
self
,
sock
,
addr
,
obj
,
'S'
,
map
=
map
)
Connection
.
__init__
(
self
,
sock
,
addr
,
obj
,
'S'
,
map
=
map
)
self
.
marshal
=
ServerMarshaller
()
self
.
decode
=
ZEO
.
zrpc
.
marshal
.
server_decode
self
.
trigger
=
ZEO
.
zrpc
.
trigger
.
trigger
(
map
)
self
.
trigger
=
ZEO
.
zrpc
.
trigger
.
trigger
(
map
)
self
.
call_from_thread
=
self
.
trigger
.
pull_trigger
self
.
call_from_thread
=
self
.
trigger
.
pull_trigger
...
@@ -591,6 +620,15 @@ class ManagedServerConnection(Connection):
...
@@ -591,6 +620,15 @@ class ManagedServerConnection(Connection):
t
.
setDaemon
(
True
)
t
.
setDaemon
(
True
)
t
.
start
()
t
.
start
()
# self.profile = cProfile.Profile()
# def message_input(self, message):
# self.profile.enable()
# try:
# Connection.message_input(self, message)
# finally:
# self.profile.disable()
def
handshake
(
self
):
def
handshake
(
self
):
# Send the server's preferred protocol to the client.
# Send the server's preferred protocol to the client.
self
.
message_output
(
self
.
current_protocol
)
self
.
message_output
(
self
.
current_protocol
)
...
@@ -602,6 +640,7 @@ class ManagedServerConnection(Connection):
...
@@ -602,6 +640,7 @@ class ManagedServerConnection(Connection):
def
close
(
self
):
def
close
(
self
):
self
.
obj
.
notifyDisconnected
()
self
.
obj
.
notifyDisconnected
()
Connection
.
close
(
self
)
Connection
.
close
(
self
)
# self.profile.dump_stats(str(time.time())+'.stats')
def
send_reply
(
self
,
msgid
,
ret
,
immediately
=
True
):
def
send_reply
(
self
,
msgid
,
ret
,
immediately
=
True
):
# encode() can pass on a wide variety of exceptions from cPickle.
# encode() can pass on a wide variety of exceptions from cPickle.
...
@@ -609,14 +648,14 @@ class ManagedServerConnection(Connection):
...
@@ -609,14 +648,14 @@ class ManagedServerConnection(Connection):
# it's acceptable -- we really do want to catch every exception
# it's acceptable -- we really do want to catch every exception
# cPickle may raise.
# cPickle may raise.
try
:
try
:
msg
=
self
.
marshal
.
encode
(
msgid
,
0
,
REPLY
,
ret
)
msg
=
self
.
encode
(
msgid
,
0
,
REPLY
,
ret
)
except
:
# see above
except
:
# see above
try
:
try
:
r
=
short_repr
(
ret
)
r
=
short_repr
(
ret
)
except
:
except
:
r
=
"<unreprable>"
r
=
"<unreprable>"
err
=
ZRPCError
(
"Couldn't pickle return %.100s"
%
r
)
err
=
ZRPCError
(
"Couldn't pickle return %.100s"
%
r
)
msg
=
self
.
marshal
.
encode
(
msgid
,
0
,
REPLY
,
(
ZRPCError
,
err
))
msg
=
self
.
encode
(
msgid
,
0
,
REPLY
,
(
ZRPCError
,
err
))
self
.
message_output
(
msg
)
self
.
message_output
(
msg
)
if
immediately
:
if
immediately
:
self
.
poll
()
self
.
poll
()
...
...
src/ZEO/zrpc/marshal.py
View file @
6d107ff8
...
@@ -11,60 +11,65 @@
...
@@ -11,60 +11,65 @@
# FOR A PARTICULAR PURPOSE
# FOR A PARTICULAR PURPOSE
#
#
##############################################################################
##############################################################################
import
cPickle
from
cPickle
import
Unpickler
,
Pickler
from
cStringIO
import
StringIO
from
cStringIO
import
StringIO
import
logging
import
logging
from
ZEO.zrpc.error
import
ZRPCError
from
ZEO.zrpc.error
import
ZRPCError
from
ZEO.zrpc.log
import
log
,
short_repr
from
ZEO.zrpc.log
import
log
,
short_repr
class
Marshaller
:
def
encode
(
*
args
):
# args: (msgid, flags, name, args)
"""Marshal requests and replies to second across network"""
# (We used to have a global pickler, but that's not thread-safe. :-( )
def
encode
(
self
,
msgid
,
flags
,
name
,
args
):
# It's not thread safe if, in the couse of pickling, we call the
"""Returns an encoded message"""
# Python interpeter, which releases the GIL.
# (We used to have a global pickler, but that's not thread-safe. :-( )
# Note that args may contain very large binary pickles already; for
# Note that args may contain very large binary pickles already; for
# this reason, it's important to use proto 1 (or higher) pickles here
# this reason, it's important to use proto 1 (or higher) pickles here
# too. For a long time, this used proto 0 pickles, and that can
# too. For a long time, this used proto 0 pickles, and that can
# bloat our pickle to 4x the size (due to high-bit and control bytes
# bloat our pickle to 4x the size (due to high-bit and control bytes
# being represented by \xij escapes in proto 0).
# being represented by \xij escapes in proto 0).
# Undocumented: cPickle.Pickler accepts a lone protocol argument;
# Undocumented: cPickle.Pickler accepts a lone protocol argument;
# pickle.py does not.
# pickle.py does not.
pickler
=
cPickle
.
Pickler
(
1
)
pickler
=
Pickler
(
1
)
pickler
.
fast
=
1
pickler
.
fast
=
1
return
pickler
.
dump
(
args
,
1
)
# Undocumented: pickler.dump(), for a cPickle.Pickler, takes
# an optional boolean argument. When true, it returns the pickle;
# when false or unspecified, it returns the pickler object itself.
@
apply
# pickle.py does none of this.
def
fast_encode
():
return
pickler
.
dump
((
msgid
,
flags
,
name
,
args
),
1
)
# Only use in cases where you *know* the data contains only basic
# Python objects
def
decode
(
self
,
msg
):
pickler
=
Pickler
(
1
)
"""Decodes msg and returns its parts"""
pickler
.
fast
=
1
unpickler
=
cPickle
.
Unpickler
(
StringIO
(
msg
))
dump
=
pickler
.
dump
unpickler
.
find_global
=
find_global
def
fast_encode
(
*
args
):
return
dump
(
args
,
1
)
try
:
return
fast_encode
return
unpickler
.
load
()
# msgid, flags, name, args
except
:
def
decode
(
msg
):
log
(
"can't decode message: %s"
%
short_repr
(
msg
),
"""Decodes msg and returns its parts"""
level
=
logging
.
ERROR
)
unpickler
=
Unpickler
(
StringIO
(
msg
))
raise
unpickler
.
find_global
=
find_global
class
ServerMarshaller
(
Marshaller
):
try
:
return
unpickler
.
load
()
# msgid, flags, name, args
def
decode
(
self
,
msg
):
except
:
"""Decodes msg and returns its parts"""
log
(
"can't decode message: %s"
%
short_repr
(
msg
),
unpickler
=
cPickle
.
Unpickler
(
StringIO
(
msg
))
level
=
logging
.
ERROR
)
unpickler
.
find_global
=
server_find_global
raise
try
:
def
server_decode
(
msg
):
return
unpickler
.
load
()
# msgid, flags, name, args
"""Decodes msg and returns its parts"""
except
:
unpickler
=
Unpickler
(
StringIO
(
msg
))
log
(
"can't decode message: %s"
%
short_repr
(
msg
),
unpickler
.
find_global
=
server_find_global
level
=
logging
.
ERROR
)
raise
try
:
return
unpickler
.
load
()
# msgid, flags, name, args
except
:
log
(
"can't decode message: %s"
%
short_repr
(
msg
),
level
=
logging
.
ERROR
)
raise
_globals
=
globals
()
_globals
=
globals
()
_silly
=
(
'__doc__'
,)
_silly
=
(
'__doc__'
,)
...
...
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