Browse Source

compact map fix bug when iterating

pull/809/head
Chris Lu 6 years ago
parent
commit
36d13355bb
  1. 68
      weed/storage/needle/compact_map.go
  2. 19
      weed/storage/needle/compact_map_test.go
  3. 2
      weed/storage/volume_vacuum_test.go

68
weed/storage/needle/compact_map.go

@ -105,24 +105,16 @@ func (cs *CompactSection) Get(key NeedleId) (*NeedleValue, bool) {
return nil, false return nil, false
} }
func (cs *CompactSection) binarySearchValues(key SectionalNeedleId) int { func (cs *CompactSection) binarySearchValues(key SectionalNeedleId) int {
l, h := 0, cs.counter-1
if h >= 0 && cs.values[h].Key < key {
return -2
}
//println("looking for key", key)
for l <= h {
m := (l + h) / 2
//println("mid", m, "key", cs.values[m].Key, cs.values[m].Offset, cs.values[m].Size)
if cs.values[m].Key < key {
l = m + 1
} else if key < cs.values[m].Key {
h = m - 1
} else {
//println("found", m)
return m
x := sort.Search(cs.counter, func(i int) bool {
return cs.values[i].Key >= key
})
if x == cs.counter {
return -1
} }
if cs.values[x].Key > key {
return -2
} }
return -1
return x
} }
//This map assumes mostly inserting increasing keys //This map assumes mostly inserting increasing keys
@ -138,21 +130,24 @@ func NewCompactMap() *CompactMap {
func (cm *CompactMap) Set(key NeedleId, offset Offset, size uint32) (oldOffset Offset, oldSize uint32) { func (cm *CompactMap) Set(key NeedleId, offset Offset, size uint32) (oldOffset Offset, oldSize uint32) {
x := cm.binarySearchCompactSection(key) x := cm.binarySearchCompactSection(key)
if x < 0 || (key-cm.list[x].start) > SectionalNeedleIdLimit { if x < 0 || (key-cm.list[x].start) > SectionalNeedleIdLimit {
//println(x, "creating", len(cm.list), "section, starting", key)
// println(x, "adding to existing", len(cm.list), "sections, starting", key)
cs := NewCompactSection(key) cs := NewCompactSection(key)
cm.list = append(cm.list, cs) cm.list = append(cm.list, cs)
x = len(cm.list) - 1 x = len(cm.list) - 1
//keep compact section sorted by start //keep compact section sorted by start
for x > 0 {
if cm.list[x-1].start > key {
for x >= 0 {
if x > 0 && cm.list[x-1].start > key {
cm.list[x] = cm.list[x-1] cm.list[x] = cm.list[x-1]
// println("shift", x, "start", cs.start, "to", x-1)
x = x - 1 x = x - 1
} else { } else {
cm.list[x] = cs cm.list[x] = cs
// println("cs", x, "start", cs.start)
break break
} }
} }
} }
// println(key, "set to section[", x, "].start", cm.list[x].start)
return cm.list[x].Set(key, offset, size) return cm.list[x].Set(key, offset, size)
} }
func (cm *CompactMap) Delete(key NeedleId) uint32 { func (cm *CompactMap) Delete(key NeedleId) uint32 {
@ -170,29 +165,19 @@ func (cm *CompactMap) Get(key NeedleId) (*NeedleValue, bool) {
return cm.list[x].Get(key) return cm.list[x].Get(key)
} }
func (cm *CompactMap) binarySearchCompactSection(key NeedleId) int { func (cm *CompactMap) binarySearchCompactSection(key NeedleId) int {
l, h := 0, len(cm.list)-1
if h < 0 {
return -5
}
if cm.list[h].start <= key {
if cm.list[h].counter < batch || key <= cm.list[h].end {
return h
}
return -4
}
for l <= h {
m := (l + h) / 2
if key < cm.list[m].start {
h = m - 1
} else { // cm.list[m].start <= key
if cm.list[m+1].start <= key {
l = m + 1
} else {
return m
if len(cm.list) == 0 {
return -1
} }
x := sort.Search(len(cm.list), func(i int) bool {
return cm.list[i].start >= key
})
if len(cm.list) == x {
return -1
} }
if cm.list[x].start == key {
return x
} }
return -3
return x - 1
} }
// Visit visits all entries or stop if any error when visiting // Visit visits all entries or stop if any error when visiting
@ -205,7 +190,10 @@ func (cm *CompactMap) Visit(visit func(NeedleValue) error) error {
return err return err
} }
} }
for _, v := range cs.values {
for i, v := range cs.values {
if i >= cs.counter {
break
}
if _, found := cs.overflow.findOverflowEntry(v.Key); !found { if _, found := cs.overflow.findOverflowEntry(v.Key); !found {
if err := visit(v.toNeedleValue(cs)); err != nil { if err := visit(v.toNeedleValue(cs)); err != nil {
cs.RUnlock() cs.RUnlock()

19
weed/storage/needle/compact_map_test.go

@ -5,6 +5,25 @@ import (
"testing" "testing"
) )
func TestOverflow2(t *testing.T) {
m := NewCompactMap()
m.Set(NeedleId(150088), 8, 3000073)
m.Set(NeedleId(150073), 8, 3000073)
m.Set(NeedleId(150089), 8, 3000073)
m.Set(NeedleId(150076), 8, 3000073)
m.Set(NeedleId(150124), 8, 3000073)
m.Set(NeedleId(150137), 8, 3000073)
m.Set(NeedleId(150147), 8, 3000073)
m.Set(NeedleId(150145), 8, 3000073)
m.Set(NeedleId(150158), 8, 3000073)
m.Set(NeedleId(150162), 8, 3000073)
m.Visit(func(value NeedleValue) error {
println("needle key:", value.Key)
return nil
})
}
func TestIssue52(t *testing.T) { func TestIssue52(t *testing.T) {
m := NewCompactMap() m := NewCompactMap()
m.Set(NeedleId(10002), 10002, 10002) m.Set(NeedleId(10002), 10002, 10002)

2
weed/storage/volume_vacuum_test.go

@ -131,7 +131,7 @@ func doSomeWritesDeletes(i int, v *Volume, t *testing.T, infos []*needleInfo) {
crc: n.Checksum, crc: n.Checksum,
} }
// println("written file", i, "checksum", n.Checksum.Value(), "size", size) // println("written file", i, "checksum", n.Checksum.Value(), "size", size)
if rand.Float64() < 0.5 {
if rand.Float64() < 0.03 {
toBeDeleted := rand.Intn(i) + 1 toBeDeleted := rand.Intn(i) + 1
oldNeedle := newEmptyNeedle(uint64(toBeDeleted)) oldNeedle := newEmptyNeedle(uint64(toBeDeleted))
v.deleteNeedle(oldNeedle) v.deleteNeedle(oldNeedle)

Loading…
Cancel
Save