Skip to content

Commit

Permalink
Merge pull request #165 from kubescape/alreadyexists
Browse files Browse the repository at this point in the history
fail Create() if object already exists
  • Loading branch information
matthyx authored Nov 25, 2024
2 parents 2714080 + d7aed14 commit 5ba7d7b
Show file tree
Hide file tree
Showing 3 changed files with 71 additions and 15 deletions.
19 changes: 18 additions & 1 deletion pkg/registry/file/storage.go
Original file line number Diff line number Diff line change
Expand Up @@ -193,7 +193,7 @@ func (s *StorageImpl) writeFiles(key string, obj runtime.Object, metaOut runtime
return nil
}

// Create adds a new object at a key even when it already exists. 'ttl' is time-to-live
// Create adds a new object at a key unless it already exists. 'ttl' is time-to-live
// in seconds (and is ignored). If no error is returned and out is not nil, out will be
// set to the read value from database.
func (s *StorageImpl) Create(ctx context.Context, key string, obj, metaOut runtime.Object, _ uint64) error {
Expand All @@ -204,6 +204,10 @@ func (s *StorageImpl) Create(ctx context.Context, key string, obj, metaOut runti
s.locks.Lock(key)
defer s.locks.Unlock(key)
spanLock.End()
// check if object already exists
if _, err := s.appFs.Stat(makePayloadPath(filepath.Join(s.root, key))); err == nil {
return storage.NewKeyExistsError(key, 0)
}
// resourceVersion should not be set on create
if version, err := s.versioner.ObjectResourceVersion(obj); err == nil && version != 0 {
msg := "resourceVersion should not be set on objects to be created"
Expand Down Expand Up @@ -292,6 +296,19 @@ func (s *StorageImpl) Get(ctx context.Context, key string, opts storage.GetOptio
// get is a helper function for Get to allow calls without locks from other methods that already have them
func (s *StorageImpl) get(ctx context.Context, key string, opts storage.GetOptions, objPtr runtime.Object) error {
p := filepath.Join(s.root, key)
if opts.ResourceVersion == "metadata" {
// get metadata from SQLite
conn, err := s.pool.Take(context.Background())
if err != nil {
return fmt.Errorf("take connection: %w", err)
}
defer s.pool.Put(conn)
metadata, err := ReadMetadata(conn, key)
if err != nil {
return fmt.Errorf("read metadata: %w", err)
}
return json.Unmarshal(metadata, objPtr)
}
payloadFile, err := s.appFs.OpenFile(makePayloadPath(p), syscall.O_DIRECT|os.O_RDONLY, 0)
if err != nil {
if errors.Is(err, afero.ErrFileNotFound) {
Expand Down
65 changes: 52 additions & 13 deletions pkg/registry/file/storage_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@ import (
"bytes"
"context"
"encoding/gob"
"encoding/json"
"fmt"
"testing"

Expand Down Expand Up @@ -264,24 +265,37 @@ func isNotFoundError(_ assert.TestingT, err error, _ ...any) bool {
func TestStorageImpl_Get(t *testing.T) {
var emptyObj bytes.Buffer
_ = gob.NewEncoder(&emptyObj).Encode(v1beta1.SBOMSyft{})
var realMeta bytes.Buffer
_ = json.NewEncoder(&realMeta).Encode(v1beta1.SBOMSyft{
ObjectMeta: v1.ObjectMeta{
Name: "toto",
},
})
var realObj bytes.Buffer
_ = gob.NewEncoder(&realObj).Encode(v1beta1.SBOMSyft{
ObjectMeta: v1.ObjectMeta{
Name: "toto",
},
Spec: v1beta1.SBOMSyftSpec{
Metadata: v1beta1.SPDXMeta{
Tool: v1beta1.ToolMeta{
Name: "syft"},
},
},
})
type args struct {
key string
opts storage.GetOptions
objPtr runtime.Object
}
tests := []struct {
name string
args args
content string
create bool
wantErr assert.ErrorAssertionFunc
want runtime.Object
name string
args args
content []byte
contentMeta []byte
create bool
wantErr assert.ErrorAssertionFunc
want runtime.Object
}{
{
name: "not found",
Expand All @@ -305,7 +319,7 @@ func TestStorageImpl_Get(t *testing.T) {
key: "/spdx.softwarecomposition.kubescape.io/sbomsyfts/kubescape/toto",
objPtr: &v1beta1.SBOMSyft{},
},
content: emptyObj.String(),
content: emptyObj.Bytes(),
create: true,
wantErr: assert.NoError,
want: &v1beta1.SBOMSyft{},
Expand All @@ -316,9 +330,31 @@ func TestStorageImpl_Get(t *testing.T) {
key: "/spdx.softwarecomposition.kubescape.io/sbomsyfts/kubescape/toto",
objPtr: &v1beta1.SBOMSyft{},
},
content: realObj.String(),
content: realObj.Bytes(),
create: true,
wantErr: assert.NoError,
want: &v1beta1.SBOMSyft{
ObjectMeta: v1.ObjectMeta{
Name: "toto",
},
Spec: v1beta1.SBOMSyftSpec{
Metadata: v1beta1.SPDXMeta{
Tool: v1beta1.ToolMeta{
Name: "syft"},
},
},
},
},
{
name: "real object - metadata only",
args: args{
key: "/spdx.softwarecomposition.kubescape.io/sbomsyfts/kubescape/toto",
objPtr: &v1beta1.SBOMSyft{},
opts: storage.GetOptions{ResourceVersion: "metadata"},
},
contentMeta: realMeta.Bytes(),
create: true,
wantErr: assert.NoError,
want: &v1beta1.SBOMSyft{
ObjectMeta: v1.ObjectMeta{
Name: "toto",
Expand All @@ -331,24 +367,27 @@ func TestStorageImpl_Get(t *testing.T) {
key: "/spdx.softwarecomposition.kubescape.io/sbomsyfts/kubescape/toto",
objPtr: &v1beta1.SBOMSyft{},
},
content: string(realObj.Bytes()[10]),
content: realObj.Bytes()[10:10],
create: true,
wantErr: isNotFoundError,
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
fs := afero.NewMemMapFs()
if tt.create {
path := getStoredPayloadFilepath(DefaultStorageRoot, tt.args.key)
_ = afero.WriteFile(fs, path, []byte(tt.content), 0644)
}
pool := NewTestPool(t.TempDir())
require.NotNil(t, pool)
defer func(pool *sqlitemigration.Pool) {
_ = pool.Close()
}(pool)
s := NewStorageImpl(fs, DefaultStorageRoot, pool, nil)
if tt.create {
conn, err := pool.Take(context.Background())
require.NoError(t, err)
require.NoError(t, WriteJSON(conn, tt.args.key, tt.contentMeta))
require.NoError(t, afero.WriteFile(fs, getStoredPayloadFilepath(DefaultStorageRoot, tt.args.key), tt.content, 0644))
pool.Put(conn)
}
if err := s.Get(context.TODO(), tt.args.key, tt.args.opts, tt.args.objPtr); !tt.wantErr(t, err) {
t.Errorf("Get() error = %v, wantErr %v", err, tt.wantErr(t, err))
}
Expand Down
2 changes: 1 addition & 1 deletion pkg/registry/file/vulnerabilitysummarystorage_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -443,9 +443,9 @@ func TestVulnSummaryStorageImpl_Get(t *testing.T) {
}(pool)
sch := scheme.Scheme
require.NoError(t, softwarecomposition.AddToScheme(sch))
realStorage := NewStorageImpl(afero.NewMemMapFs(), "/", pool, sch)

for _, tt := range tests {
realStorage := NewStorageImpl(afero.NewMemMapFs(), "/", pool, sch)
t.Run(tt.name, func(t *testing.T) {
if tt.createObj {
for i := range tt.args.keyCreatedObj {
Expand Down

0 comments on commit 5ba7d7b

Please sign in to comment.