Skip to content
GitLab
Explore
Sign in
Register
Primary navigation
Search or go to…
Project
P
picodata
Manage
Activity
Members
Labels
Plan
Issues
Issue boards
Milestones
Code
Merge requests
Repository
Branches
Commits
Tags
Repository graph
Compare revisions
Snippets
Build
Pipelines
Jobs
Pipeline schedules
Artifacts
Deploy
Container Registry
Model registry
Analyze
Value stream analytics
Contributor analytics
CI/CD analytics
Repository analytics
Model experiments
Help
Help
Support
GitLab documentation
Compare GitLab plans
Community forum
Contribute to GitLab
Provide feedback
Keyboard shortcuts
?
Snippets
Groups
Projects
Show more breadcrumbs
core
picodata
Commits
9c873a8d
Commit
9c873a8d
authored
1 year ago
by
Georgy Moshkin
Browse files
Options
Downloads
Patches
Plain Diff
refactor: extract prepare_for_snapshot function
parent
2ff0085a
No related branches found
No related tags found
No related merge requests found
Changes
1
Hide whitespace changes
Inline
Side-by-side
Showing
1 changed file
src/traft/node.rs
+89
-77
89 additions, 77 deletions
src/traft/node.rs
with
89 additions
and
77 deletions
src/traft/node.rs
+
89
−
77
View file @
9c873a8d
...
...
@@ -49,7 +49,6 @@ use ::raft::Error as RaftError;
use
::
raft
::
StateRole
as
RaftStateRole
;
use
::
raft
::
StorageError
;
use
::
raft
::
INVALID_ID
;
use
::
tarantool
::
error
::
TarantoolError
;
use
::
tarantool
::
fiber
;
use
::
tarantool
::
fiber
::
mutex
::
MutexGuard
;
use
::
tarantool
::
fiber
::
r
#
async
::
timeout
::
Error
as
TimeoutError
;
...
...
@@ -1356,6 +1355,90 @@ impl NodeImpl {
Ok
(())
}
/// Prepare for applying the raft snapshot if it's not empty.
///
/// This includes:
/// - Verifying snapshot version against global & local ones;
/// - Waiting until tarantool replication proceeds if this is a read-only replica;
/// - Fetching the rest of the snashot chunks if the first one is not the only one;
fn
prepare_for_snapshot
(
&
self
,
snapshot
:
&
raft
::
Snapshot
,
)
->
traft
::
Result
<
Option
<
SnapshotData
>>
{
if
snapshot
.is_empty
()
{
return
Ok
(
None
);
}
let
snapshot_data
=
crate
::
unwrap_ok_or!
(
SnapshotData
::
decode
(
snapshot
.get_data
()),
Err
(
e
)
=>
{
tlog!
(
Warning
,
"skipping snapshot, which failed to deserialize: {e}"
);
return
Err
(
e
.into
());
}
);
let
v_snapshot
=
snapshot_data
.schema_version
;
loop
{
let
v_local
=
local_schema_version
()
.expect
(
"storage souldn't fail"
);
let
v_global
=
self
.storage
.properties
.global_schema_version
()
.expect
(
"storage shouldn't fail"
);
#[rustfmt::skip]
debug_assert!
(
v_global
<=
v_local
,
"global schema version is only ever increased after local"
);
#[rustfmt::skip]
debug_assert!
(
v_global
<=
v_snapshot
,
"global schema version updates are distributed via raft"
);
if
v_local
>
v_snapshot
{
tlog!
(
Warning
,
"skipping stale snapshot: local schema version: {}, snapshot schema version: {}"
,
v_local
,
snapshot_data
.schema_version
,
);
return
Ok
(
None
);
}
if
!
self
.is_readonly
()
{
// Replicaset leader applies the schema changes directly.
break
;
}
if
v_local
==
v_snapshot
{
// Replicaset follower has synced schema with the leader,
// now global space dumps should be handled.
break
;
}
tlog!
(
Debug
,
"v_local: {v_local}, v_snapshot: {v_snapshot}"
);
self
.main_loop_status
(
"awaiting replication"
);
// Replicaset follower needs to sync with leader via tarantool
// replication.
let
timeout
=
MainLoop
::
TICK
*
4
;
fiber
::
sleep
(
timeout
);
}
let
mut
snapshot_data
=
snapshot_data
;
if
snapshot_data
.next_chunk_position
.is_some
()
{
self
.main_loop_status
(
"receiving snapshot"
);
let
entry_id
=
RaftEntryId
{
index
:
snapshot
.get_metadata
()
.index
,
term
:
snapshot
.get_metadata
()
.term
,
};
if
let
Err
(
e
)
=
self
.fetch_chunkwise_snapshot
(
&
mut
snapshot_data
,
entry_id
)
{
// Error has been logged.
tlog!
(
Warning
,
"dropping snapshot data"
);
return
Err
(
e
);
}
}
Ok
(
Some
(
snapshot_data
))
}
#[inline(always)]
fn
main_loop_status
(
&
self
,
status
:
&
'static
str
)
{
tlog!
(
Debug
,
"main_loop_status = '{status}'"
);
...
...
@@ -1430,83 +1513,12 @@ impl NodeImpl {
// This is a snapshot, we need to apply the snapshot at first.
let
snapshot
=
ready
.snapshot
();
let
snapshot_data
=
(||
->
Option
<
SnapshotData
>
{
if
snapshot
.is_empty
()
{
return
None
;
}
let
snapshot_data
=
crate
::
unwrap_ok_or!
(
SnapshotData
::
decode
(
snapshot
.get_data
()),
Err
(
e
)
=>
{
tlog!
(
Warning
,
"skipping snapshot, which failed to deserialize: {e}"
);
return
None
;
}
);
let
v_snapshot
=
snapshot_data
.schema_version
;
loop
{
let
v_local
=
local_schema_version
()
.expect
(
"storage souldn't fail"
);
let
v_global
=
self
.storage
.properties
.global_schema_version
()
.expect
(
"storage shouldn't fail"
);
assert!
(
v_global
<=
v_local
,
"global schema version is only ever increased after local"
);
assert!
(
v_global
<=
v_snapshot
,
"global schema version updates are distributed via raft"
);
if
v_local
>
v_snapshot
{
tlog!
(
Warning
,
"skipping stale snapshot: local schema version: {}, snapshot schema version: {}"
,
v_local
,
snapshot_data
.schema_version
,
);
return
None
;
}
if
!
self
.is_readonly
()
{
// Replicaset leader applies the schema changes directly.
return
Some
(
snapshot_data
);
}
if
v_local
==
v_snapshot
{
// Replicaset follower has synced schema with the leader,
// now global space dumps should be handled.
return
Some
(
snapshot_data
);
}
tlog!
(
Debug
,
"v_local: {v_local}, v_snapshot: {v_snapshot}"
);
self
.main_loop_status
(
"awaiting replication"
);
// Replicaset follower needs to sync with leader via tarantool
// replication.
let
timeout
=
MainLoop
::
TICK
*
4
;
fiber
::
sleep
(
timeout
);
}
})();
if
let
Some
(
mut
snapshot_data
)
=
snapshot_data
{
if
snapshot_data
.next_chunk_position
.is_some
()
{
self
.main_loop_status
(
"receiving snapshot"
);
let
entry_id
=
RaftEntryId
{
index
:
snapshot
.get_metadata
()
.index
,
term
:
snapshot
.get_metadata
()
.term
,
};
if
let
Err
(
e
)
=
self
.fetch_chunkwise_snapshot
(
&
mut
snapshot_data
,
entry_id
)
{
// Error has been logged.
_
=
e
;
tlog!
(
Warning
,
"dropping snapshot data"
);
return
;
}
}
let
Ok
(
snapshot_data
)
=
self
.prepare_for_snapshot
(
snapshot
)
else
{
// Error was already logged
return
;
};
if
let
Some
(
snapshot_data
)
=
snapshot_data
{
self
.main_loop_status
(
"applying snapshot"
);
if
let
Err
(
e
)
=
transaction
(||
->
traft
::
Result
<
()
>
{
...
...
This diff is collapsed.
Click to expand it.
Preview
0%
Loading
Try again
or
attach a new file
.
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Save comment
Cancel
Please
register
or
sign in
to comment