mirror of
https://github.com/seaweedfs/seaweedfs.git
synced 2026-05-30 05:30:23 +00:00
* fix(volume): verify the .dat-tail needle in the integrity check CheckVolumeDataIntegrity checked the last entry by file position in the .idx and, for a live needle, flipped the volume read-only when fileSize > fileTailOffset. That entry is the .dat tail only when the .idx is in append order; a key-sorted .idx (weed fix and other rebuilds listed entries by key) puts the highest-key needle last, whose tail sits mid-file, so healthy volumes went read-only on every load and re-running weed fix only reproduced the sorted index. Locate the needle at the maximum offset — the one physically last in the .dat — and verify the .dat ends exactly at it, regardless of .idx ordering. The append-ordered common case stays O(1) (the last entry's on-disk end matches the .dat size); only a key-sorted index pays a single linear scan. Deletion tombstones at the tail are now verified too, instead of skipping the file-size check. * fix(command): weed fix rebuilds the .idx in .dat offset order SaveToIdx wrote entries via AscendingVisit — sorted by key, the .sdx/.ecx shape — so the rebuilt .idx put the highest-key needle last instead of the .dat-tail needle, and dropped tombstones whose live needle was gone. Collect the live and deleted entries, sort by .dat offset, and write them in append order so the .idx stays a faithful log whose last entry is the real .dat tail.
76 lines
2.4 KiB
Go
76 lines
2.4 KiB
Go
package command
|
|
|
|
import (
|
|
"os"
|
|
"path/filepath"
|
|
"testing"
|
|
|
|
"github.com/seaweedfs/seaweedfs/weed/storage/idx"
|
|
"github.com/seaweedfs/seaweedfs/weed/storage/needle_map"
|
|
"github.com/seaweedfs/seaweedfs/weed/storage/types"
|
|
)
|
|
|
|
// TestSaveToIdxWritesOffsetOrder guards that weed fix rebuilds the .idx as an
|
|
// append-ordered log (sorted by .dat offset), not sorted by key. A key-sorted
|
|
// .idx puts the highest-key needle last instead of the .dat-tail needle, which
|
|
// flipped volumes read-only on load (issue #9688). It must also carry every
|
|
// tombstone — including one whose live needle is gone — so the last entry is
|
|
// the real .dat tail.
|
|
func TestSaveToIdxWritesOffsetOrder(t *testing.T) {
|
|
nm := needle_map.NewMemDb()
|
|
defer nm.Close()
|
|
nmDeleted := needle_map.NewMemDb()
|
|
defer nmDeleted.Close()
|
|
|
|
// Live needles with high keys at low offsets (as if written first).
|
|
for _, e := range []struct {
|
|
key uint64
|
|
offset int64
|
|
}{{30, 8}, {20, 128}, {10, 256}} {
|
|
if err := nm.Set(types.Uint64ToNeedleId(e.key), types.ToOffset(e.offset), types.Size(100)); err != nil {
|
|
t.Fatalf("nm.Set: %v", err)
|
|
}
|
|
}
|
|
// A tombstone at the .dat tail whose live needle is gone; the old SaveToIdx
|
|
// dropped these because the key was absent from the live map.
|
|
if err := nmDeleted.Set(types.Uint64ToNeedleId(5), types.ToOffset(384), types.TombstoneFileSize); err != nil {
|
|
t.Fatalf("nmDeleted.Set: %v", err)
|
|
}
|
|
|
|
scanner := &VolumeFileScanner4Fix{nm: nm, nmDeleted: nmDeleted, includeDeleted: true}
|
|
|
|
idxPath := filepath.Join(t.TempDir(), "v.idx")
|
|
if err := SaveToIdx(scanner, idxPath); err != nil {
|
|
t.Fatalf("SaveToIdx: %v", err)
|
|
}
|
|
|
|
f, err := os.Open(idxPath)
|
|
if err != nil {
|
|
t.Fatalf("open idx: %v", err)
|
|
}
|
|
defer f.Close()
|
|
|
|
var offsets []int64
|
|
var lastKey types.NeedleId
|
|
if err := idx.WalkIndexFile(f, 0, func(key types.NeedleId, offset types.Offset, size types.Size) error {
|
|
offsets = append(offsets, offset.ToActualOffset())
|
|
lastKey = key
|
|
return nil
|
|
}); err != nil {
|
|
t.Fatalf("walk idx: %v", err)
|
|
}
|
|
|
|
if len(offsets) != 4 {
|
|
t.Fatalf("expected 4 entries (3 live + 1 tombstone), got %d", len(offsets))
|
|
}
|
|
for i := 1; i < len(offsets); i++ {
|
|
if offsets[i] < offsets[i-1] {
|
|
t.Fatalf("entries not in offset order: %v", offsets)
|
|
}
|
|
}
|
|
// The .dat-tail needle (the orphan tombstone at the highest offset) must be last.
|
|
if lastKey != types.Uint64ToNeedleId(5) {
|
|
t.Errorf("last entry should be the .dat-tail tombstone (key 5), got key %v", lastKey)
|
|
}
|
|
}
|