@@ -75,22 +75,9 @@ func CreateStorage(
75
75
return nil , err
76
76
}
77
77
78
- snapshot , err := sn . Load ( )
78
+ snapshot , w , st , ents , err := loadNewestAvailableSnapshot ( lg , walDir , snapDir )
79
79
if err != nil {
80
- if err == snap .ErrNoSnapshot {
81
- lg .Debugf ("No snapshot found at %s" , snapDir )
82
- } else {
83
- return nil , errors .Errorf ("failed to load snapshot: %s" , err )
84
- }
85
- } else {
86
- // snapshot found
87
- lg .Debugf ("Loaded snapshot at Term %d and Index %d, Nodes: %+v" ,
88
- snapshot .Metadata .Term , snapshot .Metadata .Index , snapshot .Metadata .ConfState .Nodes )
89
- }
90
-
91
- w , st , ents , err := createOrReadWAL (lg , walDir , snapshot )
92
- if err != nil {
93
- return nil , errors .Errorf ("failed to create or read WAL: %s" , err )
80
+ return nil , errors .Errorf ("Failed to load snapshot and WAL: %s" , err )
94
81
}
95
82
96
83
if snapshot != nil {
@@ -120,26 +107,11 @@ func CreateStorage(
120
107
// ListSnapshots returns a list of RaftIndex of snapshots stored on disk.
121
108
// If a file is corrupted, rename the file.
122
109
func ListSnapshots (logger * flogging.FabricLogger , snapDir string ) []uint64 {
123
- dir , err := os . Open ( snapDir )
110
+ snapfiles , err := listSnapshotFiles ( logger , snapDir )
124
111
if err != nil {
125
- logger .Errorf ("Failed to open snapshot directory %s: %s" , snapDir , err )
112
+ logger .Errorf ("Failed to list snapshot files from %s: %s" , snapDir , err )
126
113
return nil
127
114
}
128
- defer dir .Close ()
129
-
130
- filenames , err := dir .Readdirnames (- 1 )
131
- if err != nil {
132
- logger .Errorf ("Failed to read snapshot files: %s" , err )
133
- return nil
134
- }
135
-
136
- snapfiles := []string {}
137
- for i := range filenames {
138
- if strings .HasSuffix (filenames [i ], ".snap" ) {
139
- snapfiles = append (snapfiles , filenames [i ])
140
- }
141
- }
142
- sort .Strings (snapfiles )
143
115
144
116
var snapshots []uint64
145
117
for _ , snapfile := range snapfiles {
@@ -242,15 +214,17 @@ func (rs *RaftStorage) Snapshot() raftpb.Snapshot {
242
214
243
215
// Store persists etcd/raft data
244
216
func (rs * RaftStorage ) Store (entries []raftpb.Entry , hardstate raftpb.HardState , snapshot raftpb.Snapshot ) error {
245
- if err := rs .wal .Save (hardstate , entries ); err != nil {
246
- return err
247
- }
248
-
249
217
if ! raft .IsEmptySnap (snapshot ) {
250
218
if err := rs .saveSnap (snapshot ); err != nil {
251
219
return err
252
220
}
221
+ }
222
+
223
+ if err := rs .wal .Save (hardstate , entries ); err != nil {
224
+ return err
225
+ }
253
226
227
+ if ! raft .IsEmptySnap (snapshot ) {
254
228
if err := rs .ram .ApplySnapshot (snapshot ); err != nil {
255
229
if err == raft .ErrSnapOutOfDate {
256
230
rs .lg .Warnf ("Attempted to apply out-of-date snapshot at Term %d and Index %d" ,
@@ -447,3 +421,54 @@ func (rs *RaftStorage) Close() error {
447
421
448
422
return nil
449
423
}
424
+
425
+ func loadNewestAvailableSnapshot (lg * flogging.FabricLogger , walDir , snapDir string ) (* raftpb.Snapshot , * wal.WAL , raftpb.HardState , []raftpb.Entry , error ) {
426
+ snapfiles , err := listSnapshotFiles (lg , snapDir )
427
+ if err != nil {
428
+ lg .Errorf ("Failed to list snapshot files from %s: %s" , snapDir , err )
429
+ }
430
+ for i := len (snapfiles ) - 1 ; i >= 0 ; i -- {
431
+ snapshot , err := snap .Read (lg .Zap (), filepath .Join (snapDir , snapfiles [i ]))
432
+ if err != nil {
433
+ lg .Warnf ("Can not read snapshot from %s: %s" , snapfiles [i ], err )
434
+ continue
435
+ }
436
+ w , st , ents , err := createOrReadWAL (lg , walDir , snapshot )
437
+ if err != nil {
438
+ lg .Warnf ("Create or read wal error: %s" , err )
439
+ continue
440
+ }
441
+ if snapshot .Metadata .Index <= st .Commit {
442
+ return snapshot , w , st , ents , nil
443
+ }
444
+ if err := w .Close (); err != nil {
445
+ return nil , nil , raftpb.HardState {}, nil , err
446
+ }
447
+ }
448
+ lg .Warnf ("Not available snapshot found in %s" , snapDir )
449
+ w , st , ents , err := createOrReadWAL (lg , walDir , nil )
450
+ return nil , w , st , ents , err
451
+ }
452
+
453
+ func listSnapshotFiles (logging * flogging.FabricLogger , snapDir string ) ([]string , error ) {
454
+ dir , err := os .Open (snapDir )
455
+ if err != nil {
456
+ return nil , err
457
+ }
458
+ defer dir .Close ()
459
+
460
+ filenames , err := dir .Readdirnames (- 1 )
461
+ if err != nil {
462
+ return nil , err
463
+ }
464
+
465
+ snapfiles := []string {}
466
+ for i := range filenames {
467
+ if strings .HasSuffix (filenames [i ], ".snap" ) {
468
+ snapfiles = append (snapfiles , filenames [i ])
469
+ }
470
+ }
471
+ sort .Strings (snapfiles )
472
+
473
+ return snapfiles , nil
474
+ }
0 commit comments