Skip to content

Commit

Permalink
Merge pull request #607 from nekomeowww/dev/binary-update
Browse files Browse the repository at this point in the history
fix: direct `[]byte` write when updating resources
  • Loading branch information
Iceber authored Nov 28, 2023
2 parents ff23b61 + 2e5a879 commit b725360
Show file tree
Hide file tree
Showing 4 changed files with 129 additions and 1 deletion.
3 changes: 2 additions & 1 deletion pkg/storage/internalstorage/resource_storage.go
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@ import (
"reflect"
"strconv"

"gorm.io/datatypes"
"gorm.io/gorm"
apierrors "k8s.io/apimachinery/pkg/api/errors"
"k8s.io/apimachinery/pkg/api/meta"
Expand Down Expand Up @@ -107,7 +108,7 @@ func (s *ResourceStorage) Update(ctx context.Context, cluster string, obj runtim
"owner_uid": ownerUID,
"uid": metaobj.GetUID(),
"resource_version": metaobj.GetResourceVersion(),
"object": buffer.Bytes(),
"object": datatypes.JSON(buffer.Bytes()),
}
if deletedAt := metaobj.GetDeletionTimestamp(); deletedAt != nil {
updatedResource["deleted_at"] = sql.NullTime{Time: deletedAt.Time, Valid: true}
Expand Down
105 changes: 105 additions & 0 deletions pkg/storage/internalstorage/resource_storage_test.go
Original file line number Diff line number Diff line change
@@ -1,15 +1,22 @@
package internalstorage

import (
"bytes"
"context"
"fmt"
"testing"

"github.com/stretchr/testify/assert"
"github.com/stretchr/testify/require"
"gorm.io/gorm"
appsv1 "k8s.io/api/apps/v1"
v1 "k8s.io/api/core/v1"
"k8s.io/apimachinery/pkg/api/meta"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
"k8s.io/apimachinery/pkg/runtime/schema"

internal "github.com/clusterpedia-io/api/clusterpedia"
"github.com/clusterpedia-io/clusterpedia/pkg/storageconfig"
)

func testApplyListOptionsToResourceQuery(t *testing.T, name string, options *internal.ListOptions, expected expected) {
Expand Down Expand Up @@ -358,6 +365,104 @@ func TestResourceStorage_deleteObject(t *testing.T) {
}
}

func TestResourceStorage_Update(t *testing.T) {
assert := assert.New(t)
require := require.New(t)

db, cleanup, err := newSQLiteDB()
require.NoError(err)
require.NotNil(db)
defer cleanup()

rs := newTestResourceStorage(db, appsv1.SchemeGroupVersion.WithResource("deployments"))

factory := storageconfig.NewStorageConfigFactory()
require.NotNil(factory)

config, err := factory.NewLegacyResourceConfig(schema.GroupResource{Group: appsv1.SchemeGroupVersion.Group, Resource: "deployments"}, true)
require.NoError(err)
require.NotNil(config)

rs.codec = config.Codec
trueRef := true

obj := &appsv1.Deployment{
TypeMeta: metav1.TypeMeta{
Kind: "Deployment",
},
ObjectMeta: metav1.ObjectMeta{
Name: "foo",
Namespace: "foobar",
OwnerReferences: []metav1.OwnerReference{
{UID: "fooer-id", Name: "fooer", Controller: &trueRef},
},
},
Spec: appsv1.DeploymentSpec{
Template: v1.PodTemplateSpec{
ObjectMeta: metav1.ObjectMeta{
Labels: map[string]string{
"foo": "bar",
},
},
},
},
}

metaObj, err := meta.Accessor(obj)
require.NoError(err)
ownerRef := metaObj.GetOwnerReferences()
require.Len(ownerRef, 1)

var buffer bytes.Buffer
err = rs.codec.Encode(obj, &buffer)
require.NoError(err)

owner := metav1.GetControllerOfNoCopy(metaObj)
require.NotNil(owner)

ownerUID := owner.UID
require.NotEmpty(ownerUID)

clusterName := "test"

err = rs.Create(context.Background(), clusterName, obj)
require.NoError(err)

var resourcesAfterCreation []Resource
err = db.
Where(Resource{
Cluster: clusterName,
Name: "foo",
Namespace: "foobar",
}).
Find(&resourcesAfterCreation).
Error
require.NoError(err)
require.Len(resourcesAfterCreation, 1)
assert.NotEmpty(resourcesAfterCreation[0].Object)

obj.Spec.Template.ObjectMeta.Labels = map[string]string{
"foo2": "bar2",
}

err = rs.Update(context.Background(), clusterName, obj)
require.NoError(err)

var resourcesAfterUpdates []Resource
err = db.
Where(Resource{
Cluster: clusterName,
Name: "foo",
Namespace: "foobar",
}).
Find(&resourcesAfterUpdates).
Error
require.NoError(err)
require.Len(resourcesAfterUpdates, 1)
assert.NotEmpty(resourcesAfterUpdates[0].Object)
assert.NotEqual(resourcesAfterUpdates[0].Object, resourcesAfterCreation[0].Object)
}

func newTestResourceStorage(db *gorm.DB, storageGVK schema.GroupVersionResource) *ResourceStorage {
return &ResourceStorage{
db: db,
Expand Down
20 changes: 20 additions & 0 deletions pkg/storage/internalstorage/storage_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@ import (
"github.com/DATA-DOG/go-sqlmock"
gmysql "gorm.io/driver/mysql"
gpostgres "gorm.io/driver/postgres"
gsqlite "gorm.io/driver/sqlite"
"gorm.io/gorm"
)

Expand All @@ -34,6 +35,25 @@ func newMockedPostgresDB() (*gorm.DB, sqlmock.Sqlmock, error) {
return gormDB, mock, nil
}

func newSQLiteDB() (*gorm.DB, func(), error) {
db, err := gorm.Open(gsqlite.Open("test.db"))
if err != nil {
return nil, func() {}, err
}

err = db.AutoMigrate(&Resource{})
if err != nil {
return nil, func() {}, err
}

return db, func() {
err := os.Remove("test.db")
if err != nil {
panic(err)
}
}, nil
}

func newMockedMySQLDB(version string) (*gorm.DB, sqlmock.Sqlmock, error) {
mockedDB, mock, err := sqlmock.New()
if err != nil {
Expand Down
2 changes: 2 additions & 0 deletions pkg/storage/internalstorage/util_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -121,6 +121,7 @@ func assertPostgresDatabaseExecutedSQL[P any](
db, mock, err := newMockedPostgresDB()
require.NoError(t, err)
require.NotNil(t, db)
require.NotNil(t, mock)

assertDatabaseExecutedSQL(t, db, mock, options, applyFn, expectedQuery, args, expectedError)
}
Expand All @@ -137,6 +138,7 @@ func assertMySQLDatabaseExecutedSQL[P any](
db, mock, err := newMockedMySQLDB(version)
require.NoError(t, err)
require.NotNil(t, db)
require.NotNil(t, mock)

assertDatabaseExecutedSQL(t, db, mock, options, applyFn, expectedQuery, args, expectedError)
}
Expand Down

0 comments on commit b725360

Please sign in to comment.