Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

add: no namespace walk support to demux snapshotter #656

Merged
merged 1 commit into from
Jun 1, 2022
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
4 changes: 4 additions & 0 deletions snapshotter/demux/cache/cache.go
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,7 @@ package cache
import (
"context"

"github.com/containerd/containerd/snapshots"
"github.com/firecracker-microvm/firecracker-containerd/snapshotter/demux/proxy"
)

Expand All @@ -28,6 +29,9 @@ type Cache interface {
// fetch function if the snapshotter is not currently cached.
Get(ctx context.Context, key string, fetch SnapshotterProvider) (*proxy.RemoteSnapshotter, error)

// WalkAll applies the provided function across all cached snapshotters.
WalkAll(ctx context.Context, fn snapshots.WalkFunc, filters ...string) error

// Closes the snapshotter and removes it from the cache.
Evict(key string) error

Expand Down
16 changes: 16 additions & 0 deletions snapshotter/demux/cache/snapshotter_cache.go
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,7 @@ import (
"fmt"
"sync"

"github.com/containerd/containerd/snapshots"
"github.com/firecracker-microvm/firecracker-containerd/snapshotter/demux/proxy"
"github.com/hashicorp/go-multierror"
)
Expand Down Expand Up @@ -56,6 +57,21 @@ func (cache *SnapshotterCache) Get(ctx context.Context, key string, fetch Snapsh
return snapshotter, nil
}

// WalkAll applies the provided function to all cached snapshotters.
func (cache *SnapshotterCache) WalkAll(ctx context.Context, fn snapshots.WalkFunc, filters ...string) error {
cache.mutex.RLock()
defer cache.mutex.RUnlock()

var allErr error

for namespace, snapshotter := range cache.snapshotters {
if err := snapshotter.Walk(ctx, fn, filters...); err != nil {
allErr = multierror.Append(allErr, fmt.Errorf("failed to walk function on snapshotter[%s]: %w", namespace, err))
}
}
return allErr
}

// Evict removes a cached snapshotter for a given key.
func (cache *SnapshotterCache) Evict(key string) error {
cache.mutex.RLock()
Expand Down
79 changes: 71 additions & 8 deletions snapshotter/demux/cache/snapshotter_cache_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,7 @@ import (
"context"
"testing"

"github.com/containerd/containerd/snapshots"
"github.com/hashicorp/go-multierror"
"github.com/pkg/errors"

Expand Down Expand Up @@ -55,13 +56,53 @@ func getCachedSnapshotter(uut *SnapshotterCache) error {
return nil
}

func getSnapshotterPropogatesErrors(uut *SnapshotterCache) error {
func getSnapshotterPropagatesErrors(uut *SnapshotterCache) error {
if _, err := uut.Get(context.Background(), "SnapshotterKey", getSnapshotterErrorFunction); err == nil {
return errors.New("Get function did not propagate errors from snapshotter generator function")
}
return nil
}

func successfulWalk(ctx context.Context, info snapshots.Info) error {
return nil
}

func applyWalkFunctionOnEmptyCache(uut *SnapshotterCache) error {
if err := uut.WalkAll(context.Background(), successfulWalk); err != nil {
return errors.New("WalkAll on empty cache incorrectly resulted in error")
}
return nil
}

func applyWalkFunctionToAllCachedSnapshotters(uut *SnapshotterCache) error {
if _, err := uut.Get(context.Background(), "Snapshotter-A", getSnapshotterOkFunction); err != nil {
return errors.Wrap(err, "Adding snapshotter A to empty cache incorrectly resulted in error")
}
if _, err := uut.Get(context.Background(), "Snapshotter-B", getSnapshotterOkFunction); err != nil {
return errors.Wrap(err, "Adding snapshotter B to empty cache incorrectly resulted in error")
}
if err := uut.WalkAll(context.Background(), successfulWalk); err != nil {
return errors.New("WalkAll on populated cache incorrectly resulted in error")
}
return nil
}

func applyWalkFunctionPropagatesErrors(uut *SnapshotterCache) error {
if _, err := uut.Get(context.Background(), "Snapshotter-A", getFailingSnapshotterOkFunction); err != nil {
return errors.Wrap(err, "Adding snapshotter A to empty cache incorrectly resulted in error")
}
// The failing snapshotter mock will fail all Walk calls before applying
// the snapshots.WalkFunc, but for the purposes of this test that is fine.
// In which case, any function will do.
walkFunc := func(ctx context.Context, info snapshots.Info) error {
return nil
}
if err := uut.WalkAll(context.Background(), walkFunc); err == nil {
return errors.New("WalkAll did not propagate errors from walk function")
}
return nil
}

func evictSnapshotterFromEmptyCache(uut *SnapshotterCache) error {
if err := uut.Evict("SnapshotterKey"); err == nil {
return errors.New("Evict function did not return error on call on empty cache")
Expand All @@ -80,7 +121,7 @@ func evictSnapshotterFromCache(uut *SnapshotterCache) error {
return nil
}

func evictSnapshotterFromCachePropogatesCloseError(uut *SnapshotterCache) error {
func evictSnapshotterFromCachePropagatesCloseError(uut *SnapshotterCache) error {
if _, err := uut.Get(context.Background(), "SnapshotterKey", getFailingSnapshotterOkFunction); err != nil {
return errors.Wrap(err, "Adding snapshotter to empty cache incorrectly resulted in error")
}
Expand Down Expand Up @@ -156,14 +197,36 @@ func TestGetSnapshotterFromCache(t *testing.T) {
}{
{"AddSnapshotterToCache", getSnapshotterFromEmptyCache},
{"GetCachedSnapshotter", getCachedSnapshotter},
{"PropogateFetchSnapshotterErrors", getSnapshotterPropogatesErrors},
{"PropogateFetchSnapshotterErrors", getSnapshotterPropagatesErrors},
}

for _, test := range tests {
t.Run(test.name, func(t *testing.T) {
uut := NewSnapshotterCache()
if err := test.run(uut); err != nil {
t.Fatalf("%s: %s", test.name, err.Error())
}
})
}
}

func TestWalkAllFunctionOnCache(t *testing.T) {
t.Parallel()

tests := []struct {
name string
run func(*SnapshotterCache) error
}{
{"ApplyWalkFunctionOnEmptyCache", applyWalkFunctionOnEmptyCache},
{"ApplyWalkFunctionToAllCachedSnapshotters", applyWalkFunctionToAllCachedSnapshotters},
{"ApplyWalkFunctionPropogatesErrors", applyWalkFunctionPropagatesErrors},
}

for _, test := range tests {
t.Run(test.name, func(t *testing.T) {
uut := NewSnapshotterCache()
if err := test.run(uut); err != nil {
t.Fatal(test.name + ": " + err.Error())
t.Fatalf("%s: %s", test.name, err.Error())
}
})
}
Expand All @@ -178,14 +241,14 @@ func TestEvictSnapshotterFromCache(t *testing.T) {
}{
{"EvictSnapshotterFromEmptyCache", evictSnapshotterFromEmptyCache},
{"EvictSnapshotterFromCache", evictSnapshotterFromCache},
{"PropogateEvictSnapshotterCloseErrors", evictSnapshotterFromCachePropogatesCloseError},
{"PropogateEvictSnapshotterCloseErrors", evictSnapshotterFromCachePropagatesCloseError},
}

for _, test := range tests {
t.Run(test.name, func(t *testing.T) {
uut := NewSnapshotterCache()
if err := test.run(uut); err != nil {
t.Fatal(test.name + ": " + err.Error())
t.Fatalf("%s: %s", test.name, err.Error())
}
})
}
Expand All @@ -207,7 +270,7 @@ func TestCloseCache(t *testing.T) {
t.Run(test.name, func(t *testing.T) {
uut := NewSnapshotterCache()
if err := test.run(uut); err != nil {
t.Fatal(test.name + ": " + err.Error())
t.Fatalf("%s: %s", test.name, err.Error())
}
})
}
Expand All @@ -228,7 +291,7 @@ func TestListSnapshotters(t *testing.T) {
t.Run(test.name, func(t *testing.T) {
uut := NewSnapshotterCache()
if err := test.run(uut); err != nil {
t.Fatal(test.name + ": " + err.Error())
t.Fatalf("%s: %s", test.name, err.Error())
}
})
}
Expand Down
3 changes: 2 additions & 1 deletion snapshotter/demux/snapshotter.go
Original file line number Diff line number Diff line change
Expand Up @@ -243,7 +243,8 @@ func (s *Snapshotter) Walk(ctx context.Context, fn snapshots.WalkFunc, filters .
contextLogger := log.G(ctx).WithField("function", "Walk")
namespace, err := getNamespaceFromContext(ctx, contextLogger)
if err != nil {
return err
contextLogger.Debug("no namespace found, proxying walk function to all cached snapshotters")
return s.cache.WalkAll(ctx, fn, filters...)
}
logger := contextLogger.WithField("namespace", namespace)

Expand Down
31 changes: 26 additions & 5 deletions snapshotter/demux/snapshotter_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -73,15 +73,36 @@ func TestReturnErrorWhenCalledWithoutNamespacedContext(t *testing.T) {
{"View", func() error { _, err := uut.View(ctx, "layerKey", ""); return err }},
{"Commit", func() error { return uut.Commit(ctx, "layer1", "layerKey") }},
{"Remove", func() error { return uut.Remove(ctx, "layerKey") }},
}

for _, test := range tests {
if err := test.run(); err == nil {
t.Fatalf("%s call did not return error", test.name)
}
}
}

func TestNoErrorWhenCalledWithoutNamespacedContext(t *testing.T) {
t.Parallel()

cache := cache.NewSnapshotterCache()
ctx := logtest.WithT(context.Background(), t)

uut := NewSnapshotter(cache, fetchOkSnapshotter)

tests := []struct {
name string
run func() error
}{
{"Walk", func() error {
var callback = func(c context.Context, i snapshots.Info) error { return nil }
return uut.Walk(ctx, callback)
}},
}

for _, test := range tests {
if err := test.run(); err == nil {
t.Fatal(test.name + " call did not return error")
if err := test.run(); err != nil {
t.Fatalf("%s call returned error on no namespace execution", test.name)
}
}
}
Expand Down Expand Up @@ -116,7 +137,7 @@ func TestReturnErrorWhenSnapshotterNotFound(t *testing.T) {

for _, test := range tests {
if err := test.run(); err == nil {
t.Fatal(test.name + " call did not return error")
t.Fatalf("%s call did not return error", test.name)
}
}
}
Expand Down Expand Up @@ -153,7 +174,7 @@ func TestReturnErrorAfterProxyFunctionFailure(t *testing.T) {
for _, test := range tests {
t.Run(test.name+"ProxyFailure", func(t *testing.T) {
if err := test.run(); err == nil {
t.Fatal(test.name + " call did not return error")
t.Fatalf("%s call did not return error", test.name)
}
})
}
Expand Down Expand Up @@ -191,7 +212,7 @@ func TestNoErrorIsReturnedOnSuccessfulProxyExecution(t *testing.T) {
for _, test := range tests {
t.Run(test.name+"SuccessfulProxyCall", func(t *testing.T) {
if err := test.run(); err != nil {
t.Fatal(test.name + " call incorrectly returned an error")
t.Fatalf("%s call incorrectly returned an error", test.name)
}
})
}
Expand Down