Merge pull request #5915 from AdamKorcz/fuzz12
Fuzzing: Add fuzzers + small modifications
This commit is contained in:
commit
d193dc2b8a
@ -18,6 +18,8 @@ package fuzz
|
|||||||
import (
|
import (
|
||||||
"bytes"
|
"bytes"
|
||||||
"context"
|
"context"
|
||||||
|
"sync"
|
||||||
|
"time"
|
||||||
|
|
||||||
fuzz "github.com/AdaLogics/go-fuzz-headers"
|
fuzz "github.com/AdaLogics/go-fuzz-headers"
|
||||||
|
|
||||||
@ -33,7 +35,11 @@ const (
|
|||||||
defaultAddress = "/tmp/containerd/containerd.sock"
|
defaultAddress = "/tmp/containerd/containerd.sock"
|
||||||
)
|
)
|
||||||
|
|
||||||
func init() {
|
var (
|
||||||
|
initDaemon sync.Once
|
||||||
|
)
|
||||||
|
|
||||||
|
func startDaemon() {
|
||||||
args := []string{"--log-level", "debug"}
|
args := []string{"--log-level", "debug"}
|
||||||
go func() {
|
go func() {
|
||||||
// This is similar to invoking the
|
// This is similar to invoking the
|
||||||
@ -42,6 +48,7 @@ func init() {
|
|||||||
// for more info.
|
// for more info.
|
||||||
command.StartDaemonForFuzzing(args)
|
command.StartDaemonForFuzzing(args)
|
||||||
}()
|
}()
|
||||||
|
time.Sleep(time.Second * 4)
|
||||||
}
|
}
|
||||||
|
|
||||||
func fuzzContext() (context.Context, context.CancelFunc) {
|
func fuzzContext() (context.Context, context.CancelFunc) {
|
||||||
@ -51,6 +58,8 @@ func fuzzContext() (context.Context, context.CancelFunc) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func FuzzContainerdImport(data []byte) int {
|
func FuzzContainerdImport(data []byte) int {
|
||||||
|
initDaemon.Do(startDaemon)
|
||||||
|
|
||||||
client, err := containerd.New(defaultAddress)
|
client, err := containerd.New(defaultAddress)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return 0
|
return 0
|
||||||
|
406
contrib/fuzz/metadata_fuzzer.go
Normal file
406
contrib/fuzz/metadata_fuzzer.go
Normal file
@ -0,0 +1,406 @@
|
|||||||
|
// +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"
|
||||||
|
"io/ioutil"
|
||||||
|
"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 := ioutil.TempDir("", "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 {
|
||||||
|
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
|
||||||
|
}
|
||||||
|
if opType%1 == 0 {
|
||||||
|
i := images.Image{}
|
||||||
|
err := f.GenerateStruct(&i)
|
||||||
|
if err != nil {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
_, _ = store.Create(ctx, i)
|
||||||
|
} else if opType%2 == 0 {
|
||||||
|
newFs, err := f.GetString()
|
||||||
|
if err != nil {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
_, _ = store.List(ctx, newFs)
|
||||||
|
} else if opType%3 == 0 {
|
||||||
|
i := images.Image{}
|
||||||
|
err := f.GenerateStruct(&i)
|
||||||
|
if err != nil {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
_, _ = store.Update(ctx, i)
|
||||||
|
} else if opType%4 == 0 {
|
||||||
|
name, err := f.GetString()
|
||||||
|
if err != nil {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
_ = store.Delete(ctx, name)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return 1
|
||||||
|
}
|
||||||
|
|
||||||
|
func FuzzLeaseManager(data []byte) int {
|
||||||
|
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
|
||||||
|
}
|
||||||
|
if opType%1 == 0 {
|
||||||
|
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
|
||||||
|
}
|
||||||
|
} else if opType%2 == 0 {
|
||||||
|
_, _ = lm.List(ctx)
|
||||||
|
} else if opType%3 == 0 {
|
||||||
|
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
|
||||||
|
})
|
||||||
|
} else if opType%4 == 0 {
|
||||||
|
l := leases.Lease{}
|
||||||
|
err = f.GenerateStruct(&l)
|
||||||
|
if err != nil {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
_ = lm.Delete(ctx, l)
|
||||||
|
} else if opType%5 == 0 {
|
||||||
|
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)
|
||||||
|
} else if opType%6 == 0 {
|
||||||
|
l := leases.Lease{}
|
||||||
|
err := f.GenerateStruct(&l)
|
||||||
|
if err != nil {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
_, _ = lm.ListResources(ctx, l)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return 1
|
||||||
|
}
|
||||||
|
|
||||||
|
func FuzzContainerStore(data []byte) int {
|
||||||
|
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
|
||||||
|
}
|
||||||
|
if opType%1 == 0 {
|
||||||
|
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
|
||||||
|
})
|
||||||
|
} else if opType%2 == 0 {
|
||||||
|
filt, err := f.GetString()
|
||||||
|
if err != nil {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
_, _ = store.List(ctx, filt)
|
||||||
|
} else if opType%3 == 0 {
|
||||||
|
id, err := f.GetString()
|
||||||
|
if err != nil {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
_ = store.Delete(ctx, id)
|
||||||
|
} else if opType%4 == 0 {
|
||||||
|
fieldpaths, err := f.GetString()
|
||||||
|
if err != nil {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
_, _ = store.Update(ctx, c, fieldpaths)
|
||||||
|
} else if opType%5 == 0 {
|
||||||
|
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 := ioutil.TempDir("", "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 {
|
||||||
|
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
|
||||||
|
}
|
||||||
|
if opType%1 == 0 {
|
||||||
|
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)
|
||||||
|
} else if opType%2 == 0 {
|
||||||
|
info := content.Info{}
|
||||||
|
err = f.GenerateStruct(&info)
|
||||||
|
if err != nil {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
_, _ = cs.Update(ctx, info)
|
||||||
|
} else if opType%3 == 0 {
|
||||||
|
walkFn := func(info content.Info) error {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
_ = cs.Walk(ctx, walkFn)
|
||||||
|
} else if opType%4 == 0 {
|
||||||
|
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)
|
||||||
|
} else if opType%5 == 0 {
|
||||||
|
_, _ = cs.ListStatuses(ctx)
|
||||||
|
} else if opType%6 == 0 {
|
||||||
|
ref, err := f.GetString()
|
||||||
|
if err != nil {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
_, _ = cs.Status(ctx, ref)
|
||||||
|
} else if opType%7 == 0 {
|
||||||
|
ref, err := f.GetString()
|
||||||
|
if err != nil {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
_ = cs.Abort(ctx, ref)
|
||||||
|
} else if opType%8 == 0 {
|
||||||
|
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
|
||||||
|
}
|
@ -61,6 +61,11 @@ compile_go_fuzzer github.com/containerd/containerd/contrib/fuzz FuzzCSWalk fuzz_
|
|||||||
compile_go_fuzzer github.com/containerd/containerd/contrib/fuzz FuzzArchiveExport fuzz_archive_export
|
compile_go_fuzzer github.com/containerd/containerd/contrib/fuzz FuzzArchiveExport fuzz_archive_export
|
||||||
compile_go_fuzzer github.com/containerd/containerd/contrib/fuzz FuzzParseAuth fuzz_parse_auth
|
compile_go_fuzzer github.com/containerd/containerd/contrib/fuzz FuzzParseAuth fuzz_parse_auth
|
||||||
compile_go_fuzzer github.com/containerd/containerd/contrib/fuzz FuzzParseProcPIDStatus fuzz_parse_proc_pid_status
|
compile_go_fuzzer github.com/containerd/containerd/contrib/fuzz FuzzParseProcPIDStatus fuzz_parse_proc_pid_status
|
||||||
|
compile_go_fuzzer github.com/containerd/containerd/contrib/fuzz FuzzImageStore fuzz_image_store
|
||||||
|
compile_go_fuzzer github.com/containerd/containerd/contrib/fuzz FuzzLeaseManager fuzz_lease_manager
|
||||||
|
compile_go_fuzzer github.com/containerd/containerd/contrib/fuzz FuzzContainerStore fuzz_container_store
|
||||||
|
compile_go_fuzzer github.com/containerd/containerd/contrib/fuzz FuzzContentStore fuzz_content_store
|
||||||
|
|
||||||
|
|
||||||
# The below fuzzers require more setup than the fuzzers above.
|
# The below fuzzers require more setup than the fuzzers above.
|
||||||
# We need the binaries from "make".
|
# We need the binaries from "make".
|
||||||
@ -72,7 +77,7 @@ export GOARCH=amd64
|
|||||||
|
|
||||||
# Build runc
|
# Build runc
|
||||||
cd $SRC/
|
cd $SRC/
|
||||||
git clone https://github.com/opencontainers/runc
|
git clone https://github.com/opencontainers/runc --branch release-1.0
|
||||||
cd runc
|
cd runc
|
||||||
make
|
make
|
||||||
make install
|
make install
|
||||||
|
Loading…
Reference in New Issue
Block a user