Browse Source

fix error for deleted files during compaction

deletion during commit may cause trouble when make up the difference during commitCompact()
pull/667/head
Chris Lu 7 years ago
parent
commit
7f7e4e9885
  1. 2
      weed/storage/needle_read_write.go
  2. 24
      weed/storage/volume_vacuum.go
  3. 60
      weed/storage/volume_vacuum_test.go

2
weed/storage/needle_read_write.go

@ -162,7 +162,7 @@ func (n *Needle) ReadData(r *os.File, offset int64, size uint32, version Version
} }
n.ParseNeedleHeader(bytes) n.ParseNeedleHeader(bytes)
if n.Size != size { if n.Size != size {
return fmt.Errorf("File Entry Not Found. Needle %d Memory %d", n.Size, size)
return fmt.Errorf("File Entry Not Found. Needle id %d expected size %d Memory %d", n.Id, n.Size, size)
} }
switch version { switch version {
case Version1: case Version1:

24
weed/storage/volume_vacuum.go

@ -119,7 +119,7 @@ func (v *Volume) makeupDiff(newDatFileName, newIdxFileName, oldDatFileName, oldI
oldDatCompactRevision, err := fetchCompactRevisionFromDatFile(oldDatFile) oldDatCompactRevision, err := fetchCompactRevisionFromDatFile(oldDatFile)
if err != nil { if err != nil {
return
return fmt.Errorf("fetchCompactRevisionFromDatFile src %s failed: %v", oldDatFile.Name(), err)
} }
if oldDatCompactRevision != v.lastCompactRevision { if oldDatCompactRevision != v.lastCompactRevision {
return fmt.Errorf("current old dat file's compact revision %d is not the expected one %d", oldDatCompactRevision, v.lastCompactRevision) return fmt.Errorf("current old dat file's compact revision %d is not the expected one %d", oldDatCompactRevision, v.lastCompactRevision)
@ -137,6 +137,7 @@ func (v *Volume) makeupDiff(newDatFileName, newIdxFileName, oldDatFileName, oldI
return fmt.Errorf("readIndexEntry %s at offset %d failed: %v", oldIdxFileName, idx_offset, err) return fmt.Errorf("readIndexEntry %s at offset %d failed: %v", oldIdxFileName, idx_offset, err)
} }
key, offset, size := idxFileEntry(IdxEntry) key, offset, size := idxFileEntry(IdxEntry)
glog.V(0).Infof("key %d offset %d size %d", key, offset, size)
if _, found := incrementedHasUpdatedIndexEntry[key]; !found { if _, found := incrementedHasUpdatedIndexEntry[key]; !found {
incrementedHasUpdatedIndexEntry[key] = keyField{ incrementedHasUpdatedIndexEntry[key] = keyField{
offset: offset, offset: offset,
@ -145,24 +146,29 @@ func (v *Volume) makeupDiff(newDatFileName, newIdxFileName, oldDatFileName, oldI
} }
} }
if len(incrementedHasUpdatedIndexEntry) > 0 {
// no updates during commit step
if len(incrementedHasUpdatedIndexEntry) == 0 {
return nil
}
// deal with updates during commit step
var ( var (
dst, idx *os.File dst, idx *os.File
) )
if dst, err = os.OpenFile(newDatFileName, os.O_RDWR, 0644); err != nil { if dst, err = os.OpenFile(newDatFileName, os.O_RDWR, 0644); err != nil {
return
return fmt.Errorf("open dat file %s failed: %v", newDatFileName, err)
} }
defer dst.Close() defer dst.Close()
if idx, err = os.OpenFile(newIdxFileName, os.O_RDWR, 0644); err != nil { if idx, err = os.OpenFile(newIdxFileName, os.O_RDWR, 0644); err != nil {
return
return fmt.Errorf("open idx file %s failed: %v", newIdxFileName, err)
} }
defer idx.Close() defer idx.Close()
var newDatCompactRevision uint16 var newDatCompactRevision uint16
newDatCompactRevision, err = fetchCompactRevisionFromDatFile(dst) newDatCompactRevision, err = fetchCompactRevisionFromDatFile(dst)
if err != nil { if err != nil {
return
return fmt.Errorf("fetchCompactRevisionFromDatFile dst %s failed: %v", dst.Name(), err)
} }
if oldDatCompactRevision+1 != newDatCompactRevision { if oldDatCompactRevision+1 != newDatCompactRevision {
return fmt.Errorf("oldDatFile %s 's compact revision is %d while newDatFile %s 's compact revision is %d", oldDatFileName, oldDatCompactRevision, newDatFileName, newDatCompactRevision) return fmt.Errorf("oldDatFile %s 's compact revision is %d while newDatFile %s 's compact revision is %d", oldDatFileName, oldDatCompactRevision, newDatFileName, newDatCompactRevision)
@ -189,12 +195,13 @@ func (v *Volume) makeupDiff(newDatFileName, newIdxFileName, oldDatFileName, oldI
} }
//updated needle //updated needle
if incre_idx_entry.offset != 0 && incre_idx_entry.size != 0 {
if incre_idx_entry.offset != 0 && incre_idx_entry.size != 0 && incre_idx_entry.size != TombstoneFileSize {
//even the needle cache in memory is hit, the need_bytes is correct //even the needle cache in memory is hit, the need_bytes is correct
glog.V(0).Infof("file %d offset %d size %d", key, int64(incre_idx_entry.offset)*NeedlePaddingSize, incre_idx_entry.size)
var needle_bytes []byte var needle_bytes []byte
needle_bytes, err = ReadNeedleBlob(oldDatFile, int64(incre_idx_entry.offset)*NeedlePaddingSize, incre_idx_entry.size) needle_bytes, err = ReadNeedleBlob(oldDatFile, int64(incre_idx_entry.offset)*NeedlePaddingSize, incre_idx_entry.size)
if err != nil { if err != nil {
return
return fmt.Errorf("ReadNeedleBlob %s key %d offset %d size %d failed: %v", oldDatFile.Name(), key, int64(incre_idx_entry.offset)*NeedlePaddingSize, incre_idx_entry.size, err)
} }
dst.Write(needle_bytes) dst.Write(needle_bytes)
util.Uint32toBytes(idx_entry_bytes[8:12], uint32(offset/NeedlePaddingSize)) util.Uint32toBytes(idx_entry_bytes[8:12], uint32(offset/NeedlePaddingSize))
@ -205,7 +212,7 @@ func (v *Volume) makeupDiff(newDatFileName, newIdxFileName, oldDatFileName, oldI
fakeDelNeedle.Cookie = 0x12345678 fakeDelNeedle.Cookie = 0x12345678
_, _, err = fakeDelNeedle.Append(dst, v.Version()) _, _, err = fakeDelNeedle.Append(dst, v.Version())
if err != nil { if err != nil {
return
return fmt.Errorf("append deleted %d failed: %v", key, err)
} }
util.Uint32toBytes(idx_entry_bytes[8:12], uint32(0)) util.Uint32toBytes(idx_entry_bytes[8:12], uint32(0))
} }
@ -216,7 +223,6 @@ func (v *Volume) makeupDiff(newDatFileName, newIdxFileName, oldDatFileName, oldI
} }
_, err = idx.Write(idx_entry_bytes) _, err = idx.Write(idx_entry_bytes)
} }
}
return nil return nil
} }

60
weed/storage/volume_vacuum_test.go

@ -69,36 +69,21 @@ func TestCompaction(t *testing.T) {
t.Fatalf("volume creation: %v", err) t.Fatalf("volume creation: %v", err)
} }
FILE_COUNT := 234
beforeCommitFileCount := 10
afterCommitFileCount := 10
infos := make([]*needleInfo, FILE_COUNT)
infos := make([]*needleInfo, beforeCommitFileCount+afterCommitFileCount)
for i := 1; i <= FILE_COUNT; i++ {
n := newRandomNeedle(uint64(i))
size, err := v.writeNeedle(n)
if err != nil {
t.Fatalf("write file %d: %v", i, err)
}
infos[i-1] = &needleInfo{
size: size,
crc: n.Checksum,
for i := 1; i <= beforeCommitFileCount; i++ {
doSomeWritesDeletes(i, v, t, infos)
} }
println("written file", i, "checksum", n.Checksum.Value(), "size", size)
v.Compact(0)
if rand.Float64() < 0.5 {
toBeDeleted := rand.Intn(i) + 1
oldNeedle := newEmptyNeedle(uint64(toBeDeleted))
v.deleteNeedle(oldNeedle)
println("deleted file", toBeDeleted)
infos[toBeDeleted-1] = &needleInfo{
size: 0,
crc: n.Checksum,
}
}
for i := 1; i <= afterCommitFileCount; i++ {
doSomeWritesDeletes(i+beforeCommitFileCount, v, t, infos)
} }
v.Compact(0)
v.commitCompact() v.commitCompact()
v.Close() v.Close()
@ -108,7 +93,12 @@ func TestCompaction(t *testing.T) {
t.Fatalf("volume reloading: %v", err) t.Fatalf("volume reloading: %v", err)
} }
for i := 1; i <= FILE_COUNT; i++ {
for i := 1; i <= beforeCommitFileCount+afterCommitFileCount; i++ {
if infos[i-1] == nil {
t.Fatal("not found file", i)
continue
}
if infos[i-1].size == 0 { if infos[i-1].size == 0 {
continue continue
@ -129,6 +119,28 @@ func TestCompaction(t *testing.T) {
} }
} }
func doSomeWritesDeletes(i int, v *Volume, t *testing.T, infos []*needleInfo) {
n := newRandomNeedle(uint64(i))
size, err := v.writeNeedle(n)
if err != nil {
t.Fatalf("write file %d: %v", i, err)
}
infos[i-1] = &needleInfo{
size: size,
crc: n.Checksum,
}
println("written file", i, "checksum", n.Checksum.Value(), "size", size)
if rand.Float64() < 0.5 {
toBeDeleted := rand.Intn(i) + 1
oldNeedle := newEmptyNeedle(uint64(toBeDeleted))
v.deleteNeedle(oldNeedle)
println("deleted file", toBeDeleted)
infos[toBeDeleted-1] = &needleInfo{
size: 0,
crc: n.Checksum,
}
}
}
type needleInfo struct { type needleInfo struct {
size uint32 size uint32

Loading…
Cancel
Save