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
Xavier Thompson
neoppod
Commits
65f5516d
Commit
65f5516d
authored
Sep 15, 2017
by
Julien Muchembled
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
WIP: Do not send invalidations for new oids, just like ZEO
parent
49631a9f
Changes
6
Hide whitespace changes
Inline
Side-by-side
Showing
6 changed files
with
46 additions
and
29 deletions
+46
-29
neo/client/app.py
neo/client/app.py
+7
-4
neo/client/transactions.py
neo/client/transactions.py
+5
-1
neo/lib/protocol.py
neo/lib/protocol.py
+4
-1
neo/master/handlers/client.py
neo/master/handlers/client.py
+2
-8
neo/master/transactions.py
neo/master/transactions.py
+6
-5
neo/tests/threaded/test.py
neo/tests/threaded/test.py
+22
-10
No files found.
neo/client/app.py
View file @
65f5516d
...
...
@@ -647,12 +647,15 @@ class Application(ThreadedApplication):
# Call finish on master
txn_context
=
txn_container
.
pop
(
transaction
)
cache_dict
=
txn_context
.
cache_dict
checked
_list
=
[
oid
for
oid
,
data
in
cache_dict
.
iteritems
()
if
data
is
CHECKED_SERIAL
]
for
oid
in
checked
_list
:
checked
=
[
oid
for
oid
,
data
in
cache_dict
.
iteritems
()
if
data
is
CHECKED_SERIAL
]
for
oid
in
checked
:
del
cache_dict
[
oid
]
created
=
txn_context
.
created_list
modified
=
set
(
cache_dict
)
modified
.
difference_update
(
created
)
ttid
=
txn_context
.
ttid
p
=
Packets
.
AskFinishTransaction
(
ttid
,
cache_dict
,
checked_list
)
p
=
Packets
.
AskFinishTransaction
(
ttid
,
modified
,
checked
,
created
)
try
:
tid
=
self
.
_askPrimary
(
p
,
cache_dict
=
cache_dict
,
callback
=
f
)
assert
tid
...
...
neo/client/transactions.py
View file @
65f5516d
...
...
@@ -17,7 +17,7 @@
from
ZODB.POSException
import
StorageTransactionError
from
neo.lib.connection
import
ConnectionClosed
from
neo.lib.locking
import
SimpleQueue
from
neo.lib.protocol
import
Packets
from
neo.lib.protocol
import
Packets
,
ZERO_OID
from
.exception
import
NEOStorageError
@
apply
...
...
@@ -52,6 +52,8 @@ class Transaction(object):
# if the id is still known by the NodeManager.
# status: 0 -> check only, 1 -> store, 2 -> failed
self
.
involved_nodes
=
{}
# {node_id: status}
# new oids
self
.
created_list
=
[]
def
wakeup
(
self
,
conn
):
self
.
queue
.
put
((
conn
,
_WakeupPacket
,
{}))
...
...
@@ -128,6 +130,8 @@ class Transaction(object):
# would just flush it on tpc_finish. This also
# prevents memory errors for big transactions.
data
=
None
if
serial
==
ZERO_OID
:
self
.
created_list
.
append
(
oid
)
self
.
cache_dict
[
oid
]
=
data
def
nodeLost
(
self
,
app
,
uuid
):
...
...
neo/lib/protocol.py
View file @
65f5516d
...
...
@@ -22,7 +22,7 @@ from struct import Struct
# The protocol version must be increased whenever upgrading a node may require
# to upgrade other nodes. It is encoded as a 4-bytes big-endian integer and
# the high order byte 0 is different from TLS Handshake (0x16).
PROTOCOL_VERSION
=
1
PROTOCOL_VERSION
=
2
ENCODED_VERSION
=
Struct
(
'!L'
).
pack
(
PROTOCOL_VERSION
)
# Avoid memory errors on corrupted data.
...
...
@@ -867,6 +867,9 @@ class FinishTransaction(Packet):
PList
(
'checked_list'
,
POID
(
'oid'
),
),
PList
(
'created_list'
,
POID
(
'oid'
),
),
)
_answer
=
PStruct
(
'answer_information_locked'
,
...
...
neo/master/handlers/client.py
View file @
65f5516d
...
...
@@ -57,15 +57,9 @@ class ClientServiceHandler(MasterHandler):
conn
.
answer
((
Errors
.
Ack
if
app
.
tm
.
vote
(
app
,
*
args
)
else
Errors
.
IncompleteTransaction
)())
def
askFinishTransaction
(
self
,
conn
,
ttid
,
oid_list
,
checked_list
):
def
askFinishTransaction
(
self
,
conn
,
ttid
,
*
args
):
app
=
self
.
app
tid
,
node_list
=
app
.
tm
.
prepare
(
app
,
ttid
,
oid_list
,
checked_list
,
conn
.
getPeerId
(),
)
tid
,
node_list
=
app
.
tm
.
prepare
(
app
,
ttid
,
conn
.
getPeerId
(),
*
args
)
if
tid
:
p
=
Packets
.
AskLockInformation
(
ttid
,
tid
)
for
node
in
node_list
:
...
...
neo/master/transactions.py
View file @
65f5516d
...
...
@@ -91,10 +91,10 @@ class Transaction(object):
def
getOIDList
(
self
):
"""
Returns the list of OIDs
us
ed in the transaction
Returns the list of OIDs
modifi
ed in the transaction
"""
return
list
(
self
.
_oid_list
)
return
self
.
_oid_list
def
isPrepared
(
self
):
"""
...
...
@@ -348,7 +348,7 @@ class TransactionManager(EventQueue):
txn
.
_failed
=
failed
return
True
def
prepare
(
self
,
app
,
ttid
,
oid_list
,
checked_list
,
msg_id
):
def
prepare
(
self
,
app
,
ttid
,
msg_id
,
modified
,
checked
,
oid_list
):
"""
Prepare a transaction to be finished
"""
...
...
@@ -360,8 +360,9 @@ class TransactionManager(EventQueue):
return
None
,
None
ready
=
app
.
getStorageReadySet
(
txn
.
_storage_readiness
)
getPartition
=
pt
.
getPartition
oid_list
+=
modified
partition_set
=
set
(
map
(
getPartition
,
oid_list
))
partition_set
.
update
(
map
(
getPartition
,
checked
_list
))
partition_set
.
update
(
map
(
getPartition
,
checked
))
partition_set
.
add
(
getPartition
(
ttid
))
node_list
=
[]
uuid_set
=
set
()
...
...
@@ -394,7 +395,7 @@ class TransactionManager(EventQueue):
self
.
_queue
.
append
(
ttid
)
logging
.
debug
(
'Finish TXN %s for %s (was %s)'
,
dump
(
tid
),
txn
.
getNode
(),
dump
(
ttid
))
txn
.
prepare
(
tid
,
oid_list
,
uuid_set
,
msg_id
)
txn
.
prepare
(
tid
,
modified
,
uuid_set
,
msg_id
)
# check if greater and foreign OID was stored
if
oid_list
:
self
.
setLastOID
(
max
(
oid_list
))
...
...
neo/tests/threaded/test.py
View file @
65f5516d
...
...
@@ -905,11 +905,16 @@ class Test(NEOThreadedTest):
def
testExternalInvalidation
(
self
,
cluster
):
# Initialize objects
t1
,
c1
=
cluster
.
getTransaction
()
old_zodb
=
hasattr
(
c1
,
'_invalidated'
)
if
old_zodb
:
# BBB: ZODB < 5
invalidations
=
lambda
conn
:
conn
.
_invalidated
else
:
invalidations
=
lambda
conn
:
conn
.
_storage
.
_invalidations
c1
.
root
()[
'x'
]
=
x1
=
PCounter
()
c1
.
root
()[
'y'
]
=
y
=
PCounter
()
y
.
value
=
1
t1
.
commit
()
# Get pickle
of y
# Get pickle
s of 0 and 1
t1
.
begin
()
x
=
c1
.
_storage
.
load
(
x1
.
_p_oid
)[
0
]
y
=
c1
.
_storage
.
load
(
y
.
_p_oid
)[
0
]
...
...
@@ -922,6 +927,10 @@ class Test(NEOThreadedTest):
txn
=
transaction
.
Transaction
()
client
.
tpc_begin
(
None
,
txn
)
client
.
store
(
x1
.
_p_oid
,
x1
.
_p_serial
,
y
,
''
,
txn
)
# At the same time, we check that there's no invalidation sent
# for new oids.
new_oid
=
client
.
new_oid
()
client
.
store
(
new_oid
,
ZERO_TID
,
x
,
''
,
txn
)
# Delay invalidation for x
with
cluster
.
master
.
filterConnection
(
cluster
.
client
)
as
m2c
:
m2c
.
delayInvalidateObjects
()
...
...
@@ -932,12 +941,21 @@ class Test(NEOThreadedTest):
x2
=
c2
.
root
()[
'x'
]
cache
.
clear
()
# bypass cache
self
.
assertEqual
(
x2
.
value
,
0
)
self
.
assertRaises
(
POSException
.
POSKeyError
if
old_zodb
else
POSException
.
ReadConflictError
,
c2
.
get
,
new_oid
)
x2
.
_p_deactivate
()
t1
.
begin
()
# process invalidation and sync connection storage
if
old_zodb
:
self
.
assertEqual
(
c2
.
get
(
new_oid
).
value
,
0
)
else
:
self
.
assertRaises
(
POSException
.
ReadConflictError
,
c2
.
get
,
new_oid
)
self
.
assertEqual
(
x2
.
value
,
0
)
self
.
assertEqual
({
x2
.
_p_oid
},
invalidations
(
c2
))
# New testing transaction. Now we can see the last value of x.
t2
.
begin
()
self
.
assertEqual
(
x2
.
value
,
1
)
self
.
assertEqual
(
c2
.
get
(
new_oid
).
value
,
0
)
# Now test cache invalidation during a load from a storage
ll
=
LockLock
()
...
...
@@ -952,9 +970,9 @@ class Test(NEOThreadedTest):
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
#
from the storage (which is <value=1, next_tid=None>) is about
# to be processed.
# At this point, x could not be found
in the cache and the
#
result from the storage (which is <value=1, next_tid=None>)
#
is about
to be processed.
# Now modify x to receive an invalidation for it.
txn
=
transaction
.
Transaction
()
client
.
tpc_begin
(
None
,
txn
)
...
...
@@ -967,12 +985,6 @@ class Test(NEOThreadedTest):
self
.
assertEqual
(
x2
.
value
,
1
)
self
.
assertEqual
(
x1
.
value
,
0
)
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.
...
...
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