Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
N
neo
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Labels
Merge Requests
2
Merge Requests
2
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Analytics
Analytics
CI / CD
Repository
Value Stream
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Jobs
Commits
Open sidebar
Kirill Smelkov
neo
Commits
bd5e1658
Commit
bd5e1658
authored
Nov 02, 2016
by
Kirill Smelkov
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
.
parent
6c679687
Changes
7
Hide whitespace changes
Inline
Side-by-side
Showing
7 changed files
with
116 additions
and
15 deletions
+116
-15
neo/lib/protocol.py
neo/lib/protocol.py
+0
-1
neo/master/backup_app.py
neo/master/backup_app.py
+7
-4
neo/master/handlers/__init__.py
neo/master/handlers/__init__.py
+11
-1
neo/scripts/runner.py
neo/scripts/runner.py
+2
-2
neo/storage/database/manager.py
neo/storage/database/manager.py
+7
-0
neo/tests/threaded/__init__.py
neo/tests/threaded/__init__.py
+4
-3
neo/tests/threaded/testReplication.py
neo/tests/threaded/testReplication.py
+85
-4
No files found.
neo/lib/protocol.py
View file @
bd5e1658
...
...
@@ -1472,7 +1472,6 @@ class AddObject(Packet):
PTID
(
'data_serial'
),
)
# NOTE
class
Replicate
(
Packet
):
"""
Notify a storage node to replicate partitions up to given 'tid'
...
...
neo/master/backup_app.py
View file @
bd5e1658
...
...
@@ -213,7 +213,8 @@ class BackupApplication(object):
for
cell
in
pt
.
getCellList
(
offset
,
readable
=
True
):
node
=
cell
.
getNode
()
assert
node
.
isConnected
(),
node
if
cell
.
backup_tid
==
prev_tid
:
# XXX ?
if
cell
.
backup_tid
==
prev_tid
:
"""
# Let's given 4 TID t0,t1,t2,t3: if a cell is only
# modified by t0 & t3 and has all data for t0, 4 values
# are possible for its 'backup_tid' until it replicates
...
...
@@ -224,10 +225,12 @@ class BackupApplication(object):
# all partitions. t1 is wrong for the same reason.
# So we have chosen the highest one (t3 - 1).
# t2 should also work but maybe harder to implement.
cell
.
backup_tid
=
add64
(
tid
,
-
1
)
cell.backup_tid = add64(tid, -1)
# XXX wrong! (we did not yet pulled the data)
logging.debug(
"partition %u: updating backup_tid of %r to %s",
offset, cell, dump(cell.backup_tid))
"""
pass
else
:
assert
cell
.
backup_tid
<
last_max_tid
,
(
cell
.
backup_tid
,
last_max_tid
,
prev_tid
,
tid
)
...
...
@@ -240,8 +243,8 @@ class BackupApplication(object):
self
.
primary_partition_dict
[
offset
]
=
\
random
.
choice
(
node_list
)
else
:
# Partition not touched, so increase 'backup_tid' of all
NOTE
# "up-to-date" replicas, without having to replicate.
(probably relates to backup_tid=tid initial bug)
# Partition not touched, so increase 'backup_tid' of all
# "up-to-date" replicas, without having to replicate.
for
cell
in
pt
.
getCellList
(
offset
,
readable
=
True
):
if
last_max_tid
<=
cell
.
backup_tid
:
cell
.
backup_tid
=
tid
...
...
neo/master/handlers/__init__.py
View file @
bd5e1658
...
...
@@ -21,6 +21,8 @@ from neo.lib.protocol import (uuid_str, NodeTypes, NodeStates, Packets,
BrokenNodeDisallowedError
,
)
X
=
0
class
MasterHandler
(
EventHandler
):
"""This class implements a generic part of the event handlers."""
...
...
@@ -68,9 +70,17 @@ class MasterHandler(EventHandler):
def
askRecovery
(
self
,
conn
):
app
=
self
.
app
backup_tid
=
app
.
backup_tid
pt_backup_tid
=
None
if
backup_tid
:
pt_backup_tid
=
app
.
pt
.
getBackupTid
()
if
X
:
print
'MASTER askRecovery .backup_tid: %r pt.getBackupTid(): %r'
%
(
backup_tid
,
pt_backup_tid
)
conn
.
answer
(
Packets
.
AnswerRecovery
(
app
.
pt
.
getID
(),
app
.
backup_tid
and
app
.
pt
.
getBackupTid
(),
#app.backup_tid and app.pt.getBackupTid(),
backup_tid
and
pt_backup_tid
,
app
.
truncate_tid
))
def
askLastIDs
(
self
,
conn
):
...
...
neo/scripts/runner.py
View file @
bd5e1658
...
...
@@ -61,8 +61,8 @@ UNIT_TEST_MODULES = [
# 'neo.tests.client.testStorageHandler',
# 'neo.tests.client.testConnectionPool',
# light functional tests
'neo.tests.threaded.test'
,
'neo.tests.threaded.testImporter'
,
#
'neo.tests.threaded.test',
#
'neo.tests.threaded.testImporter',
'neo.tests.threaded.testReplication'
,
# 'neo.tests.threaded.testSSL',
]
...
...
neo/storage/database/manager.py
View file @
bd5e1658
...
...
@@ -21,6 +21,8 @@ from neo.lib.exception import DatabaseFailure
from
neo.lib.interfaces
import
abstract
,
requires
from
neo.lib.protocol
import
ZERO_TID
X
=
0
def
lazymethod
(
func
):
def
getter
(
self
):
cls
=
self
.
__class__
...
...
@@ -246,6 +248,10 @@ class DatabaseManager(object):
return
util
.
bin
(
self
.
getConfiguration
(
'backup_tid'
))
def
_setBackupTID
(
self
,
tid
):
if
X
:
print
print
'SET backup_tid: %r'
%
tid
import
traceback
;
traceback
.
print_stack
()
tid
=
util
.
dump
(
tid
)
logging
.
debug
(
'backup_tid = %s'
,
tid
)
return
self
.
_setConfiguration
(
'backup_tid'
,
tid
)
...
...
@@ -633,6 +639,7 @@ class DatabaseManager(object):
value
"""
# FIXME checks only from tids, not oids/objects inside transactions
@
abstract
def
checkTIDRange
(
self
,
partition
,
length
,
min_tid
,
max_tid
):
"""
...
...
neo/tests/threaded/__init__.py
View file @
bd5e1658
...
...
@@ -477,8 +477,9 @@ class LoggerThreadName(str):
return
str
.
__str__
(
self
)
# filters-out packet which are detected by filter-criterions setup with .add()
# for a packed detected tobe filtered; further pkts on same conn are always filtered
# catch & delay packets which are detected by filter-criterions setup with .add().
# for a packed detected to be filtered; further pkts on same conn are always filtered.
# delayed packets are delivered after `with ConnnectionFilter` ends.
class
ConnectionFilter
(
object
):
filtered_count
=
0
...
...
@@ -488,7 +489,7 @@ class ConnectionFilter(object):
_addPacket
=
Connection
.
_addPacket
@
contextmanager
def
__new__
(
cls
,
conn_list
=
()):
def
__new__
(
cls
,
conn_list
=
()):
# NOTE conn_list=() -> for all connections
self
=
object
.
__new__
(
cls
)
self
.
filter_dict
=
{}
self
.
conn_list
=
frozenset
(
conn_list
)
...
...
neo/tests/threaded/testReplication.py
View file @
bd5e1658
...
...
@@ -33,7 +33,6 @@ from .. import Patch
from
.
import
ConnectionFilter
,
NEOCluster
,
NEOThreadedTest
,
predictable_random
# NOTE
def
backup_test
(
partitions
=
1
,
upstream_kw
=
{},
backup_kw
=
{}):
def
decorator
(
wrapped
):
def
wrapper
(
self
):
...
...
@@ -54,7 +53,38 @@ def backup_test(partitions=1, upstream_kw={}, backup_kw={}):
return
decorator
# NOTE
# handy tool to get various ids of a cluster in tests
class
IDs
:
def
__init__
(
self
,
cluster
):
self
.
cluster
=
cluster
def
_recovery
(
self
):
return
self
.
cluster
.
neoctl
.
getRecovery
()
@
property
def
ptid
(
self
):
return
self
.
_recovery
()[
0
]
@
property
def
backup_tid
(
self
):
return
self
.
_recovery
()[
1
]
@
property
def
truncated_tid
(
self
):
return
self
.
_recovery
()[
2
]
@
property
def
last_tid
(
self
):
return
self
.
cluster
.
master
.
getLastTransaction
()
# XXX and attributes
@
property
def
cluster_state
(
self
):
return
self
.
cluster
.
neoctl
.
getClusterState
()
class
ReplicationTests
(
NEOThreadedTest
):
def
checksumPartition
(
self
,
storage
,
partition
,
max_tid
=
MAX_TID
):
...
...
@@ -95,6 +125,10 @@ class ReplicationTests(NEOThreadedTest):
importZODB
(
3
)
backup
=
NEOCluster
(
partitions
=
np
,
replicas
=
nr
-
1
,
storage_count
=
5
,
upstream
=
upstream
)
U
=
IDs
(
upstream
)
B
=
IDs
(
backup
)
# U -> B propagation
try
:
backup
.
start
()
# Initialize & catch up.
...
...
@@ -104,8 +138,12 @@ class ReplicationTests(NEOThreadedTest):
# Normal case, following upstream cluster closely.
importZODB
(
17
)
self
.
tic
()
self
.
assertEqual
(
B
.
backup_tid
,
U
.
last_tid
)
self
.
assertEqual
(
B
.
last_tid
,
U
.
last_tid
)
self
.
assertEqual
(
np
*
nr
,
self
.
checkBackup
(
backup
))
# Check that a backup cluster can be restarted.
# (U -> B propagation after restart)
finally
:
backup
.
stop
()
backup
.
reset
()
...
...
@@ -115,38 +153,78 @@ class ReplicationTests(NEOThreadedTest):
ClusterStates
.
BACKINGUP
)
importZODB
(
17
)
self
.
tic
()
self
.
assertEqual
(
B
.
backup_tid
,
U
.
last_tid
)
self
.
assertEqual
(
B
.
last_tid
,
U
.
last_tid
)
self
.
assertEqual
(
np
*
nr
,
self
.
checkBackup
(
backup
))
backup
.
neoctl
.
checkReplicas
(
check_dict
,
ZERO_TID
,
None
)
self
.
tic
()
# Stop backing up, nothing truncated.
backup
.
neoctl
.
setClusterState
(
ClusterStates
.
STOPPING_BACKUP
)
self
.
tic
()
self
.
assertEqual
(
B
.
backup_tid
,
None
)
self
.
assertEqual
(
B
.
last_tid
,
U
.
last_tid
)
self
.
assertEqual
(
np
*
nr
,
self
.
checkBackup
(
backup
))
self
.
assertEqual
(
backup
.
neoctl
.
getClusterState
(),
ClusterStates
.
RUNNING
)
finally
:
backup
.
stop
()
# U -> B propagation with Mb -> Sb (Replicate) delayed
from
neo.storage.database
import
manager
as
dbmanager
from
neo.master
import
handlers
as
mhandler
#dbmanager.X = 1
#mhandler.X = 1
def
delaySecondary
(
conn
,
packet
):
if
isinstance
(
packet
,
Packets
.
Replicate
):
tid
,
upstream_name
,
source_dict
=
packet
.
decode
()
return
not
upstream_name
and
all
(
source_dict
.
itervalues
())
print
'REPLICATE tid: %r, upstream_name: %r, source_dict: %r'
%
\
(
tid
,
upstream_name
,
source_dict
)
return
True
#return not upstream_name and all(source_dict.itervalues())
return
upstream_name
!=
""
backup
.
reset
()
try
:
backup
.
start
()
backup
.
neoctl
.
setClusterState
(
ClusterStates
.
STARTING_BACKUP
)
self
.
tic
()
u_last_tid0
=
U
.
last_tid
self
.
assertEqual
(
B
.
backup_tid
,
u_last_tid0
)
self
.
assertEqual
(
B
.
last_tid
,
u_last_tid0
)
print
print
'B.backup_tid: %r'
%
B
.
backup_tid
with
backup
.
master
.
filterConnection
(
*
backup
.
storage_list
)
as
f
:
f
.
add
(
delaySecondary
)
f
.
add
(
delaySecondary
)
# NOTE delays backup pickup Mb -> Sb
print
'B.backup_tid: %r'
%
B
.
backup_tid
while
not
f
.
filtered_count
:
print
'B.backup_tid: %r'
%
B
.
backup_tid
importZODB
(
1
)
print
'B.backup_tid: %r'
%
B
.
backup_tid
self
.
tic
()
print
'B.backup_tid: %r'
%
B
.
backup_tid
print
print
'u_last_tid0: %r'
%
u_last_tid0
print
'U.last_tid: %r'
%
U
.
last_tid
print
'B.backup_tid: %r'
%
B
.
backup_tid
self
.
assertGreater
(
U
.
last_tid
,
u_last_tid0
)
self
.
assertEqual
(
B
.
backup_tid
,
u_last_tid0
)
self
.
assertEqual
(
B
.
last_tid
,
U
.
last_tid
)
backup
.
neoctl
.
setClusterState
(
ClusterStates
.
STOPPING_BACKUP
)
self
.
tic
()
self
.
assertEqual
(
B
.
cluster_state
,
ClusterStates
.
RECOVERING
)
self
.
assertEqual
(
B
.
backup_tid
,
None
)
self
.
assertEqual
(
B
.
last_tid
,
U
.
last_tid
)
# not-yet truncated
self
.
tic
()
self
.
assertEqual
(
B
.
cluster_state
,
ClusterStates
.
RUNNING
)
self
.
assertEqual
(
B
.
backup_tid
,
None
)
self
.
assertEqual
(
B
.
last_tid
,
u_last_tid0
)
# truncated after recovery
self
.
assertEqual
(
np
*
nr
,
self
.
checkBackup
(
backup
,
max_tid
=
backup
.
master
.
getLastTransaction
()))
finally
:
backup
.
stop
()
dbmanager
.
X
=
0
mhandler
.
X
=
0
# S -> Sb (AddObject) delayed
backup
.
reset
()
try
:
backup
.
start
()
...
...
@@ -158,8 +236,11 @@ class ReplicationTests(NEOThreadedTest):
while
not
f
.
filtered_count
:
importZODB
(
1
)
self
.
tic
()
# TODO assert B.last_tid = U.last_tid
# assert B.backup_tid < U.last_tid
backup
.
neoctl
.
setClusterState
(
ClusterStates
.
STOPPING_BACKUP
)
self
.
tic
()
# TODO assert B.last_tid = B^.backup_tid ( < U.last_tid )
self
.
tic
()
self
.
assertEqual
(
np
*
nr
,
self
.
checkBackup
(
backup
,
max_tid
=
backup
.
master
.
getLastTransaction
()))
...
...
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