From 3dfd623891ec500771b57835cc833d634c5cb36c Mon Sep 17 00:00:00 2001 From: Toni Spets Date: Thu, 11 May 2023 09:56:20 +0300 Subject: [PATCH] Sync replica snapshots to previous If Litestream is restarted often with high snapshot interval it's possible for it to skip multiple snapshots in a row causing long restore times and inconsistent storage patterns. --- replica.go | 25 +++++++++++++++++++++++++ 1 file changed, 25 insertions(+) diff --git a/replica.go b/replica.go index 9906a70a..60c92c93 100644 --- a/replica.go +++ b/replica.go @@ -733,6 +733,31 @@ func (r *Replica) snapshotter(ctx context.Context) { return } + logger := r.Logger() + if pos, err := r.db.Pos(); err != nil { + logger.Error("snapshotter cannot determine generation", "error", err) + } else if !pos.IsZero() { + if snapshot, err := r.maxSnapshot(ctx, pos.Generation); err != nil { + logger.Error("snapshotter cannot determine latest snapshot", "error", err) + } else if snapshot != nil { + nextSnapshot := r.SnapshotInterval - time.Since(snapshot.CreatedAt) + if nextSnapshot < 0 { + nextSnapshot = 0 + } + + logger.Info("snapshot interval adjusted", "previous", snapshot.CreatedAt.Format(time.RFC3339), "next", nextSnapshot.String()) + + select { + case <-ctx.Done(): + return + case <-time.After(nextSnapshot): + if _, err := r.Snapshot(ctx); err != nil && err != ErrNoGeneration { + logger.Error("snapshotter error", "error", err) + } + } + } + } + ticker := time.NewTicker(r.SnapshotInterval) defer ticker.Stop()