442 lines
		
	
	
		
			9.4 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			442 lines
		
	
	
		
			9.4 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| //go:build gofuzz
 | |
| // +build gofuzz
 | |
| 
 | |
| /*
 | |
|    Copyright The containerd Authors.
 | |
|    Licensed under the Apache License, Version 2.0 (the "License");
 | |
|    you may not use this file except in compliance with the License.
 | |
|    You may obtain a copy of the License at
 | |
|        http://www.apache.org/licenses/LICENSE-2.0
 | |
|    Unless required by applicable law or agreed to in writing, software
 | |
|    distributed under the License is distributed on an "AS IS" BASIS,
 | |
|    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | |
|    See the License for the specific language governing permissions and
 | |
|    limitations under the License.
 | |
| */
 | |
| 
 | |
| package fuzz
 | |
| 
 | |
| import (
 | |
| 	"context"
 | |
| 	"fmt"
 | |
| 	"os"
 | |
| 	"path/filepath"
 | |
| 
 | |
| 	fuzz "github.com/AdaLogics/go-fuzz-headers"
 | |
| 	digest "github.com/opencontainers/go-digest"
 | |
| 	ocispec "github.com/opencontainers/image-spec/specs-go/v1"
 | |
| 	bolt "go.etcd.io/bbolt"
 | |
| 
 | |
| 	"github.com/containerd/containerd/containers"
 | |
| 	"github.com/containerd/containerd/content"
 | |
| 	"github.com/containerd/containerd/content/local"
 | |
| 	"github.com/containerd/containerd/images"
 | |
| 	"github.com/containerd/containerd/leases"
 | |
| 	"github.com/containerd/containerd/metadata"
 | |
| 	"github.com/containerd/containerd/namespaces"
 | |
| 	"github.com/containerd/containerd/snapshots"
 | |
| 	"github.com/containerd/containerd/snapshots/native"
 | |
| )
 | |
| 
 | |
| func testEnv() (context.Context, *bolt.DB, func(), error) {
 | |
| 	ctx, cancel := context.WithCancel(context.Background())
 | |
| 	ctx = namespaces.WithNamespace(ctx, "testing")
 | |
| 
 | |
| 	dirname, err := os.MkdirTemp("", "fuzz-")
 | |
| 	if err != nil {
 | |
| 		return ctx, nil, nil, err
 | |
| 	}
 | |
| 
 | |
| 	db, err := bolt.Open(filepath.Join(dirname, "meta.db"), 0644, nil)
 | |
| 	if err != nil {
 | |
| 		return ctx, nil, nil, err
 | |
| 	}
 | |
| 
 | |
| 	return ctx, db, func() {
 | |
| 		db.Close()
 | |
| 		_ = os.RemoveAll(dirname)
 | |
| 		cancel()
 | |
| 	}, nil
 | |
| }
 | |
| 
 | |
| func FuzzImageStore(data []byte) int {
 | |
| 	imageStoreOptions := map[int]string{
 | |
| 		0: "Create",
 | |
| 		1: "List",
 | |
| 		2: "Update",
 | |
| 		3: "Delete",
 | |
| 	}
 | |
| 
 | |
| 	ctx, db, cancel, err := testEnv()
 | |
| 	if err != nil {
 | |
| 		return 0
 | |
| 	}
 | |
| 	defer cancel()
 | |
| 	store := metadata.NewImageStore(metadata.NewDB(db, nil, nil))
 | |
| 	f := fuzz.NewConsumer(data)
 | |
| 	noOfOperations, err := f.GetInt()
 | |
| 	if err != nil {
 | |
| 		return 0
 | |
| 	}
 | |
| 	maxOperations := 50
 | |
| 	for i := 0; i < noOfOperations%maxOperations; i++ {
 | |
| 		opType, err := f.GetInt()
 | |
| 		if err != nil {
 | |
| 			return 0
 | |
| 		}
 | |
| 		switch imageStoreOptions[opType%len(imageStoreOptions)] {
 | |
| 		case "Create":
 | |
| 			i := images.Image{}
 | |
| 			err := f.GenerateStruct(&i)
 | |
| 			if err != nil {
 | |
| 				return 0
 | |
| 			}
 | |
| 			_, _ = store.Create(ctx, i)
 | |
| 		case "List":
 | |
| 			newFs, err := f.GetString()
 | |
| 			if err != nil {
 | |
| 				return 0
 | |
| 			}
 | |
| 			_, _ = store.List(ctx, newFs)
 | |
| 		case "Update":
 | |
| 			i := images.Image{}
 | |
| 			err := f.GenerateStruct(&i)
 | |
| 			if err != nil {
 | |
| 				return 0
 | |
| 			}
 | |
| 			_, _ = store.Update(ctx, i)
 | |
| 		case "Delete":
 | |
| 			name, err := f.GetString()
 | |
| 			if err != nil {
 | |
| 				return 0
 | |
| 			}
 | |
| 			_ = store.Delete(ctx, name)
 | |
| 		}
 | |
| 	}
 | |
| 	return 1
 | |
| }
 | |
| 
 | |
| func FuzzLeaseManager(data []byte) int {
 | |
| 	leaseManagerOptions := map[int]string{
 | |
| 		0: "Create",
 | |
| 		1: "List",
 | |
| 		2: "AddResource",
 | |
| 		3: "Delete",
 | |
| 		4: "DeleteResource",
 | |
| 		5: "ListResources",
 | |
| 	}
 | |
| 	ctx, db, cancel, err := testEnv()
 | |
| 	if err != nil {
 | |
| 		return 0
 | |
| 	}
 | |
| 	defer cancel()
 | |
| 	lm := metadata.NewLeaseManager(metadata.NewDB(db, nil, nil))
 | |
| 
 | |
| 	f := fuzz.NewConsumer(data)
 | |
| 	noOfOperations, err := f.GetInt()
 | |
| 	if err != nil {
 | |
| 		return 0
 | |
| 	}
 | |
| 	maxOperations := 50
 | |
| 	for i := 0; i < noOfOperations%maxOperations; i++ {
 | |
| 		opType, err := f.GetInt()
 | |
| 		if err != nil {
 | |
| 			return 0
 | |
| 		}
 | |
| 		switch leaseManagerOptions[opType%len(leaseManagerOptions)] {
 | |
| 		case "Create":
 | |
| 			err := db.Update(func(tx *bolt.Tx) error {
 | |
| 				sm := make(map[string]string)
 | |
| 				err2 := f.FuzzMap(&sm)
 | |
| 				if err2 != nil {
 | |
| 					return err2
 | |
| 				}
 | |
| 				_, _ = lm.Create(ctx, leases.WithLabels(sm))
 | |
| 				return nil
 | |
| 			})
 | |
| 			if err != nil {
 | |
| 				return 0
 | |
| 			}
 | |
| 		case "List":
 | |
| 			_, _ = lm.List(ctx)
 | |
| 		case "AddResource":
 | |
| 			l := leases.Lease{}
 | |
| 			err := f.GenerateStruct(&l)
 | |
| 			if err != nil {
 | |
| 				return 0
 | |
| 			}
 | |
| 			r := leases.Resource{}
 | |
| 			err = f.GenerateStruct(&r)
 | |
| 			if err != nil {
 | |
| 				return 0
 | |
| 			}
 | |
| 			db.Update(func(tx *bolt.Tx) error {
 | |
| 				_ = lm.AddResource(metadata.WithTransactionContext(ctx, tx), l, r)
 | |
| 				return nil
 | |
| 			})
 | |
| 		case "Delete":
 | |
| 			l := leases.Lease{}
 | |
| 			err = f.GenerateStruct(&l)
 | |
| 			if err != nil {
 | |
| 				return 0
 | |
| 			}
 | |
| 			_ = lm.Delete(ctx, l)
 | |
| 		case "DeleteResource":
 | |
| 			l := leases.Lease{}
 | |
| 			err := f.GenerateStruct(&l)
 | |
| 			if err != nil {
 | |
| 				return 0
 | |
| 			}
 | |
| 			r := leases.Resource{}
 | |
| 			err = f.GenerateStruct(&r)
 | |
| 			if err != nil {
 | |
| 				return 0
 | |
| 			}
 | |
| 			_ = lm.DeleteResource(ctx, l, r)
 | |
| 		case "ListResources":
 | |
| 			l := leases.Lease{}
 | |
| 			err := f.GenerateStruct(&l)
 | |
| 			if err != nil {
 | |
| 				return 0
 | |
| 			}
 | |
| 			_, _ = lm.ListResources(ctx, l)
 | |
| 		}
 | |
| 	}
 | |
| 	return 1
 | |
| }
 | |
| 
 | |
| func FuzzContainerStore(data []byte) int {
 | |
| 	containerStoreOptions := map[int]string{
 | |
| 		0: "Create",
 | |
| 		1: "List",
 | |
| 		2: "Delete",
 | |
| 		3: "Update",
 | |
| 		4: "Get",
 | |
| 	}
 | |
| 	ctx, db, cancel, err := testEnv()
 | |
| 	if err != nil {
 | |
| 		return 0
 | |
| 	}
 | |
| 	defer cancel()
 | |
| 
 | |
| 	store := metadata.NewContainerStore(metadata.NewDB(db, nil, nil))
 | |
| 	c := containers.Container{}
 | |
| 	f := fuzz.NewConsumer(data)
 | |
| 	noOfOperations, err := f.GetInt()
 | |
| 	if err != nil {
 | |
| 		return 0
 | |
| 	}
 | |
| 	maxOperations := 50
 | |
| 	for i := 0; i < noOfOperations%maxOperations; i++ {
 | |
| 		opType, err := f.GetInt()
 | |
| 		if err != nil {
 | |
| 			return 0
 | |
| 		}
 | |
| 		switch containerStoreOptions[opType%len(containerStoreOptions)] {
 | |
| 		case "Create":
 | |
| 			err := f.GenerateStruct(&c)
 | |
| 			if err != nil {
 | |
| 				return 0
 | |
| 			}
 | |
| 			db.Update(func(tx *bolt.Tx) error {
 | |
| 				_, _ = store.Create(metadata.WithTransactionContext(ctx, tx), c)
 | |
| 				return nil
 | |
| 			})
 | |
| 		case "List":
 | |
| 			filt, err := f.GetString()
 | |
| 			if err != nil {
 | |
| 				return 0
 | |
| 			}
 | |
| 			_, _ = store.List(ctx, filt)
 | |
| 		case "Delete":
 | |
| 			id, err := f.GetString()
 | |
| 			if err != nil {
 | |
| 				return 0
 | |
| 			}
 | |
| 			_ = store.Delete(ctx, id)
 | |
| 		case "Update":
 | |
| 			fieldpaths, err := f.GetString()
 | |
| 			if err != nil {
 | |
| 				return 0
 | |
| 			}
 | |
| 			_, _ = store.Update(ctx, c, fieldpaths)
 | |
| 		case "Get":
 | |
| 			id, err := f.GetString()
 | |
| 			if err != nil {
 | |
| 				return 0
 | |
| 			}
 | |
| 			_, _ = store.Get(ctx, id)
 | |
| 		}
 | |
| 	}
 | |
| 	return 1
 | |
| }
 | |
| 
 | |
| type testOptions struct {
 | |
| 	extraSnapshots map[string]func(string) (snapshots.Snapshotter, error)
 | |
| }
 | |
| 
 | |
| type testOpt func(*testOptions)
 | |
| 
 | |
| func testDB(opt ...testOpt) (context.Context, *metadata.DB, func(), error) {
 | |
| 	ctx, cancel := context.WithCancel(context.Background())
 | |
| 	ctx = namespaces.WithNamespace(ctx, "testing")
 | |
| 
 | |
| 	var topts testOptions
 | |
| 
 | |
| 	for _, o := range opt {
 | |
| 		o(&topts)
 | |
| 	}
 | |
| 
 | |
| 	dirname, err := os.MkdirTemp("", "fuzzing-")
 | |
| 	if err != nil {
 | |
| 		return ctx, nil, func() { cancel() }, err
 | |
| 	}
 | |
| 	defer os.RemoveAll(dirname)
 | |
| 
 | |
| 	snapshotter, err := native.NewSnapshotter(filepath.Join(dirname, "native"))
 | |
| 	if err != nil {
 | |
| 		return ctx, nil, func() { cancel() }, err
 | |
| 	}
 | |
| 
 | |
| 	snapshotters := map[string]snapshots.Snapshotter{
 | |
| 		"native": snapshotter,
 | |
| 	}
 | |
| 
 | |
| 	for name, fn := range topts.extraSnapshots {
 | |
| 		snapshotter, err := fn(filepath.Join(dirname, name))
 | |
| 		if err != nil {
 | |
| 			return ctx, nil, func() { cancel() }, err
 | |
| 		}
 | |
| 		snapshotters[name] = snapshotter
 | |
| 	}
 | |
| 
 | |
| 	cs, err := local.NewStore(filepath.Join(dirname, "content"))
 | |
| 	if err != nil {
 | |
| 		return ctx, nil, func() { cancel() }, err
 | |
| 	}
 | |
| 
 | |
| 	bdb, err := bolt.Open(filepath.Join(dirname, "metadata.db"), 0644, nil)
 | |
| 	if err != nil {
 | |
| 		return ctx, nil, func() { cancel() }, err
 | |
| 	}
 | |
| 
 | |
| 	db := metadata.NewDB(bdb, cs, snapshotters)
 | |
| 	if err := db.Init(ctx); err != nil {
 | |
| 		return ctx, nil, func() { cancel() }, err
 | |
| 	}
 | |
| 
 | |
| 	return ctx, db, func() {
 | |
| 		bdb.Close()
 | |
| 		if err := os.RemoveAll(dirname); err != nil {
 | |
| 			fmt.Println("Failed removing temp dir")
 | |
| 		}
 | |
| 		cancel()
 | |
| 	}, nil
 | |
| }
 | |
| 
 | |
| func FuzzContentStore(data []byte) int {
 | |
| 	contentStoreOptions := map[int]string{
 | |
| 		0: "Info",
 | |
| 		1: "Update",
 | |
| 		2: "Walk",
 | |
| 		3: "Delete",
 | |
| 		4: "ListStatuses",
 | |
| 		5: "Status",
 | |
| 		6: "Abort",
 | |
| 		7: "Commit",
 | |
| 	}
 | |
| 	ctx, db, cancel, err := testDB()
 | |
| 	defer cancel()
 | |
| 	if err != nil {
 | |
| 		return 0
 | |
| 	}
 | |
| 
 | |
| 	cs := db.ContentStore()
 | |
| 	f := fuzz.NewConsumer(data)
 | |
| 	noOfOperations, err := f.GetInt()
 | |
| 	if err != nil {
 | |
| 		return 0
 | |
| 	}
 | |
| 	maxOperations := 50
 | |
| 	for i := 0; i < noOfOperations%maxOperations; i++ {
 | |
| 		opType, err := f.GetInt()
 | |
| 		if err != nil {
 | |
| 			return 0
 | |
| 		}
 | |
| 		switch contentStoreOptions[opType%len(contentStoreOptions)] {
 | |
| 		case "Info":
 | |
| 			blob, err := f.GetBytes()
 | |
| 			if err != nil {
 | |
| 				return 0
 | |
| 			}
 | |
| 			dgst := digest.FromBytes(blob)
 | |
| 			err = dgst.Validate()
 | |
| 			if err != nil {
 | |
| 				return 0
 | |
| 			}
 | |
| 			_, _ = cs.Info(ctx, dgst)
 | |
| 		case "Update":
 | |
| 			info := content.Info{}
 | |
| 			err = f.GenerateStruct(&info)
 | |
| 			if err != nil {
 | |
| 				return 0
 | |
| 			}
 | |
| 			_, _ = cs.Update(ctx, info)
 | |
| 		case "Walk":
 | |
| 			walkFn := func(info content.Info) error {
 | |
| 				return nil
 | |
| 			}
 | |
| 			_ = cs.Walk(ctx, walkFn)
 | |
| 		case "Delete":
 | |
| 			blob, err := f.GetBytes()
 | |
| 			if err != nil {
 | |
| 				return 0
 | |
| 			}
 | |
| 			dgst := digest.FromBytes(blob)
 | |
| 			err = dgst.Validate()
 | |
| 			if err != nil {
 | |
| 				return 0
 | |
| 			}
 | |
| 			_ = cs.Delete(ctx, dgst)
 | |
| 		case "ListStatuses":
 | |
| 			_, _ = cs.ListStatuses(ctx)
 | |
| 		case "Status":
 | |
| 			ref, err := f.GetString()
 | |
| 			if err != nil {
 | |
| 				return 0
 | |
| 			}
 | |
| 			_, _ = cs.Status(ctx, ref)
 | |
| 		case "Abort":
 | |
| 			ref, err := f.GetString()
 | |
| 			if err != nil {
 | |
| 				return 0
 | |
| 			}
 | |
| 			_ = cs.Abort(ctx, ref)
 | |
| 		case "Commit":
 | |
| 			desc := ocispec.Descriptor{}
 | |
| 			err = f.GenerateStruct(&desc)
 | |
| 			if err != nil {
 | |
| 				return 0
 | |
| 			}
 | |
| 			ref, err := f.GetString()
 | |
| 			if err != nil {
 | |
| 				return 0
 | |
| 			}
 | |
| 			csWriter, err := cs.Writer(ctx,
 | |
| 				content.WithDescriptor(desc),
 | |
| 				content.WithRef(ref))
 | |
| 			if err != nil {
 | |
| 				return 0
 | |
| 			}
 | |
| 			defer csWriter.Close()
 | |
| 			p, err := f.GetBytes()
 | |
| 			if err != nil {
 | |
| 				return 0
 | |
| 			}
 | |
| 			_, _ = csWriter.Write(p)
 | |
| 			_ = csWriter.Commit(ctx, 0, csWriter.Digest())
 | |
| 		}
 | |
| 	}
 | |
| 	return 1
 | |
| }
 | 
