Metadata garbage collection
Marks and sweeps unreferenced objects. Add snapshot cleanup to metadata. Add content garbage collection Add dirty flags for snapshotters and content store which are set on deletion and used during the next garbage collection. Cleanup content store backend when content metadata is removed. Signed-off-by: Derek McGowan <derek@mcgstyle.net>
This commit is contained in:
@@ -1,11 +1,31 @@
|
||||
package metadata
|
||||
|
||||
import (
|
||||
"context"
|
||||
"encoding/binary"
|
||||
"fmt"
|
||||
"io"
|
||||
"io/ioutil"
|
||||
"math/rand"
|
||||
"os"
|
||||
"path/filepath"
|
||||
"runtime/pprof"
|
||||
"testing"
|
||||
"time"
|
||||
|
||||
"github.com/boltdb/bolt"
|
||||
"github.com/containerd/containerd/containers"
|
||||
"github.com/containerd/containerd/content"
|
||||
"github.com/containerd/containerd/content/local"
|
||||
"github.com/containerd/containerd/errdefs"
|
||||
"github.com/containerd/containerd/gc"
|
||||
"github.com/containerd/containerd/images"
|
||||
"github.com/containerd/containerd/namespaces"
|
||||
"github.com/containerd/containerd/snapshot"
|
||||
"github.com/containerd/containerd/snapshot/naive"
|
||||
"github.com/gogo/protobuf/types"
|
||||
digest "github.com/opencontainers/go-digest"
|
||||
ocispec "github.com/opencontainers/image-spec/specs-go/v1"
|
||||
"github.com/pkg/errors"
|
||||
)
|
||||
|
||||
@@ -176,3 +196,366 @@ func readDBVersion(db *bolt.DB, schema []byte) (int, error) {
|
||||
}
|
||||
return version, nil
|
||||
}
|
||||
|
||||
func TestMetadataCollector(t *testing.T) {
|
||||
mdb, cs, sn, cleanup := newStores(t)
|
||||
defer cleanup()
|
||||
|
||||
var (
|
||||
ctx = context.Background()
|
||||
|
||||
objects = []object{
|
||||
blob(bytesFor(1), true),
|
||||
blob(bytesFor(2), false),
|
||||
blob(bytesFor(3), true),
|
||||
blob(bytesFor(4), false, "containerd.io/gc.root", time.Now().String()),
|
||||
newSnapshot("1", "", false, false),
|
||||
newSnapshot("2", "1", false, false),
|
||||
newSnapshot("3", "2", false, false),
|
||||
newSnapshot("4", "3", false, false),
|
||||
newSnapshot("5", "3", false, true),
|
||||
container("1", "4"),
|
||||
image("image-1", digestFor(2)),
|
||||
}
|
||||
remaining []gc.Node
|
||||
)
|
||||
|
||||
if err := mdb.Update(func(tx *bolt.Tx) error {
|
||||
for _, obj := range objects {
|
||||
node, err := create(obj, tx, cs, sn)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
if node != nil {
|
||||
remaining = append(remaining, *node)
|
||||
}
|
||||
}
|
||||
return nil
|
||||
}); err != nil {
|
||||
t.Fatalf("Creation failed: %+v", err)
|
||||
}
|
||||
|
||||
if err := mdb.GarbageCollect(ctx); err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
|
||||
var actual []gc.Node
|
||||
|
||||
if err := mdb.View(func(tx *bolt.Tx) error {
|
||||
nodeC := make(chan gc.Node)
|
||||
var scanErr error
|
||||
go func() {
|
||||
defer close(nodeC)
|
||||
scanErr = scanAll(ctx, tx, nodeC)
|
||||
}()
|
||||
for node := range nodeC {
|
||||
actual = append(actual, node)
|
||||
}
|
||||
return scanErr
|
||||
}); err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
|
||||
checkNodesEqual(t, actual, remaining)
|
||||
}
|
||||
|
||||
func BenchmarkGarbageCollect(b *testing.B) {
|
||||
b.Run("10-Sets", benchmarkTrigger(10))
|
||||
b.Run("100-Sets", benchmarkTrigger(100))
|
||||
b.Run("1000-Sets", benchmarkTrigger(1000))
|
||||
b.Run("10000-Sets", benchmarkTrigger(10000))
|
||||
}
|
||||
|
||||
func benchmarkTrigger(n int) func(b *testing.B) {
|
||||
return func(b *testing.B) {
|
||||
mdb, cs, sn, cleanup := newStores(b)
|
||||
defer cleanup()
|
||||
|
||||
objects := []object{}
|
||||
|
||||
// TODO: Allow max to be configurable
|
||||
for i := 0; i < n; i++ {
|
||||
objects = append(objects,
|
||||
blob(bytesFor(int64(i)), false),
|
||||
image(fmt.Sprintf("image-%d", i), digestFor(int64(i))),
|
||||
)
|
||||
lastSnapshot := 6
|
||||
for j := 0; j <= lastSnapshot; j++ {
|
||||
var parent string
|
||||
key := fmt.Sprintf("snapshot-%d-%d", i, j)
|
||||
if j > 0 {
|
||||
parent = fmt.Sprintf("snapshot-%d-%d", i, j-1)
|
||||
}
|
||||
objects = append(objects, newSnapshot(key, parent, false, false))
|
||||
}
|
||||
objects = append(objects, container(fmt.Sprintf("container-%d", i), fmt.Sprintf("snapshot-%d-%d", i, lastSnapshot)))
|
||||
|
||||
}
|
||||
|
||||
// TODO: Create set of objects for removal
|
||||
|
||||
var (
|
||||
ctx = context.Background()
|
||||
|
||||
remaining []gc.Node
|
||||
)
|
||||
|
||||
if err := mdb.Update(func(tx *bolt.Tx) error {
|
||||
for _, obj := range objects {
|
||||
node, err := create(obj, tx, cs, sn)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
if node != nil {
|
||||
remaining = append(remaining, *node)
|
||||
}
|
||||
}
|
||||
return nil
|
||||
}); err != nil {
|
||||
b.Fatalf("Creation failed: %+v", err)
|
||||
}
|
||||
|
||||
// TODO: reset benchmark
|
||||
b.ResetTimer()
|
||||
//b.StopTimer()
|
||||
|
||||
labels := pprof.Labels("worker", "trigger")
|
||||
pprof.Do(ctx, labels, func(ctx context.Context) {
|
||||
for i := 0; i < b.N; i++ {
|
||||
|
||||
// TODO: Add removal objects
|
||||
|
||||
//b.StartTimer()
|
||||
|
||||
if err := mdb.GarbageCollect(ctx); err != nil {
|
||||
b.Fatal(err)
|
||||
}
|
||||
|
||||
//b.StopTimer()
|
||||
|
||||
//var actual []gc.Node
|
||||
|
||||
//if err := db.View(func(tx *bolt.Tx) error {
|
||||
// nodeC := make(chan gc.Node)
|
||||
// var scanErr error
|
||||
// go func() {
|
||||
// defer close(nodeC)
|
||||
// scanErr = scanAll(ctx, tx, nodeC)
|
||||
// }()
|
||||
// for node := range nodeC {
|
||||
// actual = append(actual, node)
|
||||
// }
|
||||
// return scanErr
|
||||
//}); err != nil {
|
||||
// t.Fatal(err)
|
||||
//}
|
||||
|
||||
//checkNodesEqual(t, actual, remaining)
|
||||
}
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
func bytesFor(i int64) []byte {
|
||||
r := rand.New(rand.NewSource(i))
|
||||
var b [256]byte
|
||||
_, err := r.Read(b[:])
|
||||
if err != nil {
|
||||
panic(err)
|
||||
}
|
||||
return b[:]
|
||||
}
|
||||
|
||||
func digestFor(i int64) digest.Digest {
|
||||
r := rand.New(rand.NewSource(i))
|
||||
dgstr := digest.SHA256.Digester()
|
||||
_, err := io.Copy(dgstr.Hash(), io.LimitReader(r, 256))
|
||||
if err != nil {
|
||||
panic(err)
|
||||
}
|
||||
return dgstr.Digest()
|
||||
}
|
||||
|
||||
type object struct {
|
||||
data interface{}
|
||||
removed bool
|
||||
labels map[string]string
|
||||
}
|
||||
|
||||
func create(obj object, tx *bolt.Tx, cs content.Store, sn snapshot.Snapshotter) (*gc.Node, error) {
|
||||
var (
|
||||
node *gc.Node
|
||||
namespace = "test"
|
||||
ctx = namespaces.WithNamespace(context.Background(), namespace)
|
||||
)
|
||||
|
||||
switch v := obj.data.(type) {
|
||||
case testContent:
|
||||
ctx := WithTransactionContext(ctx, tx)
|
||||
expected := digest.FromBytes(v.data)
|
||||
w, err := cs.Writer(ctx, "test-ref", int64(len(v.data)), expected)
|
||||
if err != nil {
|
||||
return nil, errors.Wrap(err, "failed to create writer")
|
||||
}
|
||||
if _, err := w.Write(v.data); err != nil {
|
||||
return nil, errors.Wrap(err, "write blob failed")
|
||||
}
|
||||
if err := w.Commit(ctx, int64(len(v.data)), expected, content.WithLabels(obj.labels)); err != nil {
|
||||
return nil, errors.Wrap(err, "failed to commit blob")
|
||||
}
|
||||
if !obj.removed {
|
||||
node = &gc.Node{
|
||||
Type: ResourceContent,
|
||||
Namespace: namespace,
|
||||
Key: expected.String(),
|
||||
}
|
||||
}
|
||||
case testSnapshot:
|
||||
ctx := WithTransactionContext(ctx, tx)
|
||||
if v.active {
|
||||
_, err := sn.Prepare(ctx, v.key, v.parent, snapshot.WithLabels(obj.labels))
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
} else {
|
||||
akey := fmt.Sprintf("%s-active", v.key)
|
||||
_, err := sn.Prepare(ctx, akey, v.parent)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
if err := sn.Commit(ctx, v.key, akey, snapshot.WithLabels(obj.labels)); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
}
|
||||
if !obj.removed {
|
||||
node = &gc.Node{
|
||||
Type: ResourceSnapshot,
|
||||
Namespace: namespace,
|
||||
Key: fmt.Sprintf("naive/%s", v.key),
|
||||
}
|
||||
}
|
||||
case testImage:
|
||||
image := images.Image{
|
||||
Name: v.name,
|
||||
Target: v.target,
|
||||
Labels: obj.labels,
|
||||
}
|
||||
_, err := NewImageStore(tx).Create(ctx, image)
|
||||
if err != nil {
|
||||
return nil, errors.Wrap(err, "failed to create image")
|
||||
}
|
||||
case testContainer:
|
||||
container := containers.Container{
|
||||
ID: v.id,
|
||||
SnapshotKey: v.snapshot,
|
||||
Snapshotter: "naive",
|
||||
Labels: obj.labels,
|
||||
|
||||
Runtime: containers.RuntimeInfo{
|
||||
Name: "testruntime",
|
||||
},
|
||||
Spec: &types.Any{},
|
||||
}
|
||||
_, err := NewContainerStore(tx).Create(ctx, container)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
}
|
||||
|
||||
return node, nil
|
||||
}
|
||||
|
||||
func blob(b []byte, r bool, l ...string) object {
|
||||
return object{
|
||||
data: testContent{
|
||||
data: b,
|
||||
},
|
||||
removed: r,
|
||||
labels: labelmap(l...),
|
||||
}
|
||||
}
|
||||
|
||||
func image(n string, d digest.Digest, l ...string) object {
|
||||
return object{
|
||||
data: testImage{
|
||||
name: n,
|
||||
target: ocispec.Descriptor{
|
||||
MediaType: "irrelevant",
|
||||
Digest: d,
|
||||
Size: 256,
|
||||
},
|
||||
},
|
||||
removed: false,
|
||||
labels: labelmap(l...),
|
||||
}
|
||||
}
|
||||
|
||||
func newSnapshot(key, parent string, active, r bool, l ...string) object {
|
||||
return object{
|
||||
data: testSnapshot{
|
||||
key: key,
|
||||
parent: parent,
|
||||
active: active,
|
||||
},
|
||||
removed: r,
|
||||
labels: labelmap(l...),
|
||||
}
|
||||
}
|
||||
|
||||
func container(id, s string, l ...string) object {
|
||||
return object{
|
||||
data: testContainer{
|
||||
id: id,
|
||||
snapshot: s,
|
||||
},
|
||||
removed: false,
|
||||
labels: labelmap(l...),
|
||||
}
|
||||
}
|
||||
|
||||
type testContent struct {
|
||||
data []byte
|
||||
}
|
||||
|
||||
type testSnapshot struct {
|
||||
key string
|
||||
parent string
|
||||
active bool
|
||||
}
|
||||
|
||||
type testImage struct {
|
||||
name string
|
||||
target ocispec.Descriptor
|
||||
}
|
||||
|
||||
type testContainer struct {
|
||||
id string
|
||||
snapshot string
|
||||
}
|
||||
|
||||
func newStores(t testing.TB) (*DB, content.Store, snapshot.Snapshotter, func()) {
|
||||
td, err := ioutil.TempDir("", "gc-test-")
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
db, err := bolt.Open(filepath.Join(td, "meta.db"), 0644, nil)
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
|
||||
nsn, err := naive.NewSnapshotter(filepath.Join(td, "snapshots"))
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
|
||||
lcs, err := local.NewStore(filepath.Join(td, "content"))
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
|
||||
mdb := NewDB(db, lcs, map[string]snapshot.Snapshotter{"naive": nsn})
|
||||
|
||||
return mdb, mdb.ContentStore(), mdb.Snapshotter("naive"), func() {
|
||||
os.RemoveAll(td)
|
||||
}
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user