2022-06-12 14:43:43 +02:00
|
|
|
package index
|
2020-06-23 22:13:25 +02:00
|
|
|
|
|
|
|
import (
|
|
|
|
"math/rand"
|
|
|
|
"testing"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/restic/restic/internal/restic"
|
|
|
|
rtest "github.com/restic/restic/internal/test"
|
|
|
|
)
|
|
|
|
|
|
|
|
func TestIndexMapBasic(t *testing.T) {
|
|
|
|
t.Parallel()
|
|
|
|
|
|
|
|
var (
|
|
|
|
id restic.ID
|
|
|
|
m indexMap
|
|
|
|
r = rand.New(rand.NewSource(98765))
|
|
|
|
)
|
|
|
|
|
|
|
|
for i := 1; i <= 400; i++ {
|
|
|
|
r.Read(id[:])
|
|
|
|
rtest.Assert(t, m.get(id) == nil, "%v retrieved but not added", id)
|
|
|
|
|
2022-02-13 17:24:09 +01:00
|
|
|
m.add(id, 0, 0, 0, 0)
|
2020-06-23 22:13:25 +02:00
|
|
|
rtest.Assert(t, m.get(id) != nil, "%v added but not retrieved", id)
|
|
|
|
rtest.Equals(t, uint(i), m.len())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestIndexMapForeach(t *testing.T) {
|
|
|
|
t.Parallel()
|
|
|
|
|
|
|
|
const N = 10
|
|
|
|
|
|
|
|
var m indexMap
|
|
|
|
|
|
|
|
// Don't crash on empty map.
|
|
|
|
m.foreach(func(*indexEntry) bool { return true })
|
|
|
|
|
|
|
|
for i := 0; i < N; i++ {
|
|
|
|
var id restic.ID
|
|
|
|
id[0] = byte(i)
|
2022-02-13 17:24:09 +01:00
|
|
|
m.add(id, i, uint32(i), uint32(i), uint32(i/2))
|
2020-06-23 22:13:25 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
seen := make(map[int]struct{})
|
|
|
|
m.foreach(func(e *indexEntry) bool {
|
|
|
|
i := int(e.id[0])
|
|
|
|
rtest.Assert(t, i < N, "unknown id %v in indexMap", e.id)
|
|
|
|
rtest.Equals(t, i, e.packIndex)
|
|
|
|
rtest.Equals(t, i, int(e.length))
|
|
|
|
rtest.Equals(t, i, int(e.offset))
|
2022-02-13 17:24:09 +01:00
|
|
|
rtest.Equals(t, i/2, int(e.uncompressedLength))
|
2020-06-23 22:13:25 +02:00
|
|
|
|
|
|
|
seen[i] = struct{}{}
|
|
|
|
return true
|
|
|
|
})
|
|
|
|
|
|
|
|
rtest.Equals(t, N, len(seen))
|
|
|
|
|
|
|
|
ncalls := 0
|
|
|
|
m.foreach(func(*indexEntry) bool {
|
|
|
|
ncalls++
|
|
|
|
return false
|
|
|
|
})
|
|
|
|
rtest.Equals(t, 1, ncalls)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestIndexMapForeachWithID(t *testing.T) {
|
|
|
|
t.Parallel()
|
|
|
|
|
|
|
|
const ndups = 3
|
|
|
|
|
|
|
|
var (
|
|
|
|
id restic.ID
|
|
|
|
m indexMap
|
|
|
|
r = rand.New(rand.NewSource(1234321))
|
|
|
|
)
|
|
|
|
r.Read(id[:])
|
|
|
|
|
|
|
|
// No result (and no crash) for empty map.
|
|
|
|
n := 0
|
|
|
|
m.foreachWithID(id, func(*indexEntry) { n++ })
|
|
|
|
rtest.Equals(t, 0, n)
|
|
|
|
|
|
|
|
// Test insertion and retrieval of duplicates.
|
|
|
|
for i := 0; i < ndups; i++ {
|
2022-02-13 17:24:09 +01:00
|
|
|
m.add(id, i, 0, 0, 0)
|
2020-06-23 22:13:25 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
for i := 0; i < 100; i++ {
|
|
|
|
var otherid restic.ID
|
|
|
|
r.Read(otherid[:])
|
2022-02-13 17:24:09 +01:00
|
|
|
m.add(otherid, -1, 0, 0, 0)
|
2020-06-23 22:13:25 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
n = 0
|
|
|
|
var packs [ndups]bool
|
|
|
|
m.foreachWithID(id, func(e *indexEntry) {
|
|
|
|
packs[e.packIndex] = true
|
|
|
|
n++
|
|
|
|
})
|
|
|
|
rtest.Equals(t, ndups, n)
|
|
|
|
|
|
|
|
for i := range packs {
|
|
|
|
rtest.Assert(t, packs[i], "duplicate from pack %d not retrieved", i)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-06-16 23:12:30 +02:00
|
|
|
func TestHashedArrayTree(t *testing.T) {
|
|
|
|
hat := newHAT()
|
|
|
|
const testSize = 1024
|
|
|
|
for i := uint(0); i < testSize; i++ {
|
|
|
|
rtest.Assert(t, hat.Size() == i, "expected hat size %v got %v", i, hat.Size())
|
|
|
|
e, idx := hat.Alloc()
|
|
|
|
rtest.Assert(t, idx == i, "expected entry at idx %v got %v", i, idx)
|
|
|
|
e.length = uint32(i)
|
|
|
|
}
|
|
|
|
for i := uint(0); i < testSize; i++ {
|
|
|
|
e := hat.Ref(i)
|
|
|
|
rtest.Assert(t, e.length == uint32(i), "expected entry to contain %v got %v", uint32(i), e.length)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-06-23 22:13:25 +02:00
|
|
|
func BenchmarkIndexMapHash(b *testing.B) {
|
|
|
|
var m indexMap
|
2022-02-13 17:24:09 +01:00
|
|
|
m.add(restic.ID{}, 0, 0, 0, 0) // Trigger lazy initialization.
|
2020-06-23 22:13:25 +02:00
|
|
|
|
|
|
|
ids := make([]restic.ID, 128) // 4 KiB.
|
|
|
|
r := rand.New(rand.NewSource(time.Now().UnixNano()))
|
|
|
|
for i := range ids {
|
|
|
|
r.Read(ids[i][:])
|
|
|
|
}
|
|
|
|
|
|
|
|
b.ReportAllocs()
|
|
|
|
b.SetBytes(int64(len(restic.ID{}) * len(ids)))
|
|
|
|
b.ResetTimer()
|
|
|
|
|
|
|
|
for i := 0; i < b.N; i++ {
|
|
|
|
for _, id := range ids {
|
|
|
|
m.hash(id)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|