Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
N
neoppod
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
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Analytics
Analytics
CI / CD
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
Ivan Tyagov
neoppod
Commits
591184df
Commit
591184df
authored
Jun 15, 2016
by
Julien Muchembled
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
wip (ZODB5)
parent
4dfdf05a
Changes
10
Show whitespace changes
Inline
Side-by-side
Showing
10 changed files
with
71 additions
and
51 deletions
+71
-51
neo/client/Storage.py
neo/client/Storage.py
+3
-6
neo/client/__init__.py
neo/client/__init__.py
+27
-19
neo/client/app.py
neo/client/app.py
+10
-7
neo/client/handlers/master.py
neo/client/handlers/master.py
+4
-1
neo/client/iterator.py
neo/client/iterator.py
+1
-1
neo/lib/handler.py
neo/lib/handler.py
+3
-2
neo/tests/client/testClientApp.py
neo/tests/client/testClientApp.py
+1
-0
neo/tests/functional/testClient.py
neo/tests/functional/testClient.py
+1
-1
neo/tests/threaded/test.py
neo/tests/threaded/test.py
+19
-12
neo/tests/threaded/testReplication.py
neo/tests/threaded/testReplication.py
+2
-2
No files found.
neo/client/Storage.py
View file @
591184df
...
...
@@ -159,11 +159,8 @@ class Storage(BaseStorage.BaseStorage,
except
NEOStorageNotFoundError
:
raise
POSException
.
POSKeyError
(
oid
)
def
sync
(
self
,
force
=
True
):
# XXX: sync() is part of IMVCCStorage and we don't want to be called
# from afterCompletion() so it may not be a good place to ping the
# master here. See also monkey-patch in __init__.py
self
.
app
.
lastTransaction
()
def
sync
(
self
):
return
self
.
app
.
sync
()
def
copyTransactionsFrom
(
self
,
source
,
verbose
=
False
):
""" Zope compliant API """
...
...
@@ -186,7 +183,7 @@ class Storage(BaseStorage.BaseStorage,
def
lastTransaction
(
self
):
# Used in ZODB unit tests
return
self
.
app
.
last
Transaction
()
return
self
.
app
.
last
_tid
def
_clear_temp
(
self
):
raise
NotImplementedError
...
...
neo/client/__init__.py
View file @
591184df
...
...
@@ -12,15 +12,11 @@
#
##############################################################################
import
app
# set up signal handers early enough to do it in the main thread
if
1
:
def
patch
():
from
hashlib
import
md5
from
ZODB.Connection
import
Connection
def
_check
(
f
,
*
args
):
h
=
md5
(
f
.
func_code
.
co_code
).
hexdigest
()
assert
h
in
args
,
h
H
=
lambda
f
:
md5
(
f
.
func_code
.
co_code
).
hexdigest
()
# Allow serial to be returned as late as tpc_finish
#
...
...
@@ -28,9 +24,7 @@ if 1:
# removing the requirement to serialise second commit phase (tpc_vote
# to tpc_finish/tpc_abort).
_check
(
Connection
.
tpc_finish
,
'ab9b1b8d82c40e5fffa84f7bc4ea3a8b'
,
# Python 2.7
)
h
=
H
(
Connection
.
tpc_finish
)
def
tpc_finish
(
self
,
transaction
):
"""Indicate confirmation that the transaction is done."""
...
...
@@ -61,18 +55,28 @@ if 1:
# </patch>
self
.
_tpc_cleanup
()
global
OLD_ZODB
OLD_ZODB
=
h
in
(
'ab9b1b8d82c40e5fffa84f7bc4ea3a8b'
,
# Python 2.7
)
if
OLD_ZODB
:
Connection
.
tpc_finish
=
tpc_finish
else
:
# Merged upstream.
assert
hasattr
(
Connection
,
'_warn_about_returned_serial'
)
# IStorage implementations usually need to provide a "network barrier",
# at least for NEO & ZEO, to make sure we have an up-to-date view of
# the storage. It's unclear whether sync() is a good place to do this
# because a round-trip to the server introduces latency and we prefer
# it's not done when it's not useful.
# For example, we know we are up-to-date after a successful commit,
# so this should not be done in afterCompletion(), and anyway, we don't
# know any legitimate use of DB access outside a transaction.
# sync() is used to provide a "network barrier", which is required for
# NEO & ZEO to make sure our view of the storage includes all changes done
# so far by other clients. But a round-trip to the server introduces
# latency so it must not be done when it's not useful. Note also that a
# successful commit (which ends with a response from the master) already
# acts as a "network barrier".
# BBB: What this monkey-patch does has been merged in ZODB5.
if
not
hasattr
(
Connection
,
'_flush_invalidations'
):
return
_check
(
Connection
.
afterCompletion
,
assert
H
(
Connection
.
afterCompletion
)
in
(
'cd3a080b80fd957190ff3bb867149448'
,
# Python 2.7
)
...
...
@@ -81,3 +85,7 @@ if 1:
# PATCH: do not call sync()
self
.
_flush_invalidations
()
Connection
.
afterCompletion
=
afterCompletion
patch
()
import
app
# set up signal handers early enough to do it in the main thread
neo/client/app.py
View file @
591184df
...
...
@@ -25,6 +25,7 @@ from functools import partial
from
ZODB.POSException
import
UndoError
,
StorageTransactionError
,
ConflictError
from
ZODB.POSException
import
ReadConflictError
from
ZODB.ConflictResolution
import
ResolvedSerial
from
.
import
OLD_ZODB
from
persistent.TimeStamp
import
TimeStamp
from
neo.lib
import
logging
...
...
@@ -108,7 +109,6 @@ class Application(ThreadedApplication):
self
.
_loading_oid
=
None
self
.
new_oid_list
=
()
self
.
last_oid
=
'
\
0
'
*
8
self
.
last_tid
=
None
self
.
storage_event_handler
=
storage
.
StorageEventHandler
(
self
)
self
.
storage_bootstrap_handler
=
storage
.
StorageBootstrapHandler
(
self
)
self
.
storage_handler
=
storage
.
StorageAnswersHandler
(
self
)
...
...
@@ -136,7 +136,11 @@ class Application(ThreadedApplication):
self
.
compress
=
compress
def
__getattr__
(
self
,
attr
):
if
attr
==
'pt'
:
if
attr
in
(
'last_tid'
,
'pt'
):
if
self
.
_connecting_to_master_node
.
locked
():
if
attr
==
'last_tid'
:
return
else
:
self
.
_getMasterConnection
()
return
self
.
__getattribute__
(
attr
)
...
...
@@ -609,7 +613,7 @@ class Application(ThreadedApplication):
logging
.
error
(
'tpc_store failed'
)
raise
NEOStorageError
(
'tpc_store failed'
)
elif
oid
in
resolved_oid_set
:
append
((
oid
,
ResolvedSerial
))
append
((
oid
,
ResolvedSerial
)
if
OLD_ZODB
else
oid
)
return
result
def
tpc_vote
(
self
,
transaction
,
tryToResolveConflict
):
...
...
@@ -962,9 +966,8 @@ class Application(ThreadedApplication):
from
.iterator
import
iterator
def
lastTransaction
(
self
):
self
.
_askPrimary
(
Packets
.
AskLastTransaction
())
return
self
.
last_tid
def
sync
(
self
):
self
.
_askPrimary
(
Packets
.
Ping
())
def
pack
(
self
,
t
):
tid
=
TimeStamp
(
*
time
.
gmtime
(
t
)[:
5
]
+
(
t
%
60
,
)).
raw
()
...
...
neo/client/handlers/master.py
View file @
591184df
...
...
@@ -99,7 +99,10 @@ class PrimaryNotificationsHandler(MTEventHandler):
app
=
self
.
app
ltid
=
packet
.
decode
()[
0
]
if
app
.
last_tid
!=
ltid
:
if
app
.
master_conn
is
None
:
# Either we're connecting or we already know the last tid
# via invalidations.
assert
app
.
master_conn
is
None
,
app
.
master_conn
if
1
:
app
.
_cache_lock_acquire
()
try
:
if
app
.
last_tid
<
ltid
:
...
...
neo/client/iterator.py
View file @
591184df
...
...
@@ -63,7 +63,7 @@ def iterator(app, start=None, stop=None):
"""NEO transaction iterator"""
if
start
is
None
:
start
=
ZERO_TID
stop
=
min
(
stop
or
MAX_TID
,
app
.
last
Transaction
()
)
stop
=
min
(
stop
or
MAX_TID
,
app
.
last
_tid
)
while
1
:
max_tid
,
chunk
=
app
.
transactionLog
(
start
,
stop
,
CHUNK_LENGTH
)
if
not
chunk
:
...
...
neo/lib/handler.py
View file @
591184df
...
...
@@ -226,8 +226,9 @@ class MTEventHandler(EventHandler):
def
packetReceived
(
self
,
conn
,
packet
,
kw
=
{}):
"""Redirect all received packet to dispatcher thread."""
if
packet
.
isResponse
()
and
type
(
packet
)
is
not
Packets
.
Pong
:
if
not
self
.
dispatcher
.
dispatch
(
conn
,
packet
.
getId
(),
packet
,
kw
):
if
packet
.
isResponse
():
if
not
(
self
.
dispatcher
.
dispatch
(
conn
,
packet
.
getId
(),
packet
,
kw
)
or
type
(
packet
)
is
Packets
.
Pong
):
raise
ProtocolError
(
'Unexpected response packet from %r: %r'
%
(
conn
,
packet
))
else
:
...
...
neo/tests/client/testClientApp.py
View file @
591184df
...
...
@@ -38,6 +38,7 @@ class Dispatcher(object):
def
_getMasterConnection
(
self
):
if
self
.
master_conn
is
None
:
self
.
last_tid
=
None
self
.
uuid
=
1
+
(
UUID_NAMESPACES
[
NodeTypes
.
CLIENT
]
<<
24
)
self
.
num_partitions
=
10
self
.
num_replicas
=
1
...
...
neo/tests/functional/testClient.py
View file @
591184df
...
...
@@ -335,7 +335,7 @@ class ClientTests(NEOFunctionalTest):
t3
.
description
=
'desc'
st3
.
tpc_begin
(
t3
)
# retreive the last revision
data
,
serial
=
st3
.
load
(
oid
,
''
)
data
,
serial
=
st3
.
load
(
oid
)
# try to store again, should not be delayed
st3
.
store
(
oid
,
serial
,
data
,
''
,
t3
)
# the vote should not timeout
...
...
neo/tests/threaded/test.py
View file @
591184df
...
...
@@ -630,7 +630,8 @@ class Test(NEOThreadedTest):
t
.
begin
()
s0
.
stop
()
# force client to ask s1
self
.
assertEqual
(
sorted
(
c
.
root
()),
[
1
])
t0
,
t1
=
c
.
_storage
.
iterator
()
self
.
tic
()
t0
,
t1
=
c
.
db
().
storage
.
iterator
()
finally
:
cluster
.
stop
()
...
...
@@ -885,7 +886,7 @@ class Test(NEOThreadedTest):
# Now test cache invalidation during a load from a storage
ll
=
LockLock
()
def
_loadFromStorage
(
orig
,
*
args
):
def
break_after
(
orig
,
*
args
):
try
:
return
orig
(
*
args
)
finally
:
...
...
@@ -893,7 +894,7 @@ class Test(NEOThreadedTest):
x2
.
_p_deactivate
()
# Remove last version of x from cache
cache
.
_remove
(
cache
.
_oid_dict
[
x2
.
_p_oid
].
pop
())
with
ll
,
Patch
(
cluster
.
client
,
_loadFromStorage
=
_loadFromStorage
):
with
ll
,
Patch
(
cluster
.
client
,
_loadFromStorage
=
break_after
):
t
=
self
.
newThread
(
x2
.
_p_activate
)
ll
()
# At this point, x could not be found the cache and the result
...
...
@@ -911,16 +912,18 @@ class Test(NEOThreadedTest):
self
.
assertEqual
(
x2
.
value
,
1
)
self
.
assertEqual
(
x1
.
value
,
0
)
# l1 is acquired and l2 is released
def
invalidations
(
conn
):
try
:
return
conn
.
_storage
.
_invalidations
except
AttributeError
:
# BBB: ZODB < 5
return
conn
.
_invalidated
# Change x again from 0 to 1, while the checking connection c1
# is suspended at the beginning of the transaction t1,
# between Storage.sync() and flush of invalidations.
def
_flush_invalidations
(
orig
):
ll
()
orig
()
x1
.
_p_deactivate
()
t1
.
abort
()
with
ll
,
Patch
(
c1
,
_flush_invalidations
=
_flush_invalidations
):
with
ll
,
Patch
(
c1
.
_storage
,
sync
=
break_after
):
t
=
self
.
newThread
(
t1
.
begin
)
ll
()
txn
=
transaction
.
Transaction
()
...
...
@@ -928,10 +931,14 @@ class Test(NEOThreadedTest):
client
.
store
(
x2
.
_p_oid
,
tid
,
y
,
''
,
txn
)
tid
=
client
.
tpc_finish
(
txn
,
None
)
client
.
close
()
self
.
assertEqual
(
invalidations
(
c1
),
{
x1
.
_p_oid
})
t
.
join
()
# A transaction really begins when it acquires the lock to flush
# invalidations. The previous lastTransaction() only does a ping
# to make sure we have a recent enough view of the DB.
# A transaction really begins when it gets the last tid from the
# storage, just before flushing invalidations (on ZODB < 5, it's
# when it acquires the lock to flush invalidations). The previous
# call to sync() only does a ping to make sure we have a recent
# enough view of the DB.
self
.
assertFalse
(
invalidations
(
c1
))
self
.
assertEqual
(
x1
.
value
,
1
)
finally
:
...
...
@@ -970,7 +977,7 @@ class Test(NEOThreadedTest):
# transaction before the last one, and clearing the cache before
# reloading x.
c1
.
_storage
.
load
(
x
.
_p_oid
)
t0
,
t1
,
t2
=
c1
.
_
storage
.
iterator
()
t0
,
t1
,
t2
=
c1
.
db
().
storage
.
iterator
()
self
.
assertEqual
(
map
(
u64
,
t0
.
oid_list
),
[
0
])
self
.
assertEqual
(
map
(
u64
,
t1
.
oid_list
),
[
0
,
1
])
# Check oid 1 is part of transaction metadata.
...
...
neo/tests/threaded/testReplication.py
View file @
591184df
...
...
@@ -373,14 +373,14 @@ class ReplicationTests(NEOThreadedTest):
self
.
tic
()
s1
.
stop
()
cluster
.
join
((
s1
,))
t0
,
t1
,
t2
=
c
.
_
storage
.
iterator
()
t0
,
t1
,
t2
=
c
.
db
().
storage
.
iterator
()
s1
.
resetNode
()
s1
.
start
()
self
.
tic
()
self
.
assertEqual
([],
cluster
.
getOutdatedCells
())
s0
.
stop
()
cluster
.
join
((
s0
,))
t0
,
t1
,
t2
=
c
.
_
storage
.
iterator
()
t0
,
t1
,
t2
=
c
.
db
().
storage
.
iterator
()
finally
:
cluster
.
stop
()
...
...
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