mirror of
https://github.com/rancher/steve.git
synced 2025-08-12 03:31:35 +00:00
adding create changes from issue 42788 to release v2.8
This commit is contained in:
parent
9fb3e50aa8
commit
64d5abd9eb
@ -13,15 +13,6 @@ import (
|
|||||||
"strconv"
|
"strconv"
|
||||||
|
|
||||||
"github.com/pkg/errors"
|
"github.com/pkg/errors"
|
||||||
"github.com/rancher/apiserver/pkg/types"
|
|
||||||
"github.com/rancher/steve/pkg/accesscontrol"
|
|
||||||
"github.com/rancher/steve/pkg/attributes"
|
|
||||||
metricsStore "github.com/rancher/steve/pkg/stores/metrics"
|
|
||||||
"github.com/rancher/steve/pkg/stores/partition"
|
|
||||||
"github.com/rancher/wrangler/v2/pkg/data"
|
|
||||||
corecontrollers "github.com/rancher/wrangler/v2/pkg/generated/controllers/core/v1"
|
|
||||||
"github.com/rancher/wrangler/v2/pkg/schemas/validation"
|
|
||||||
"github.com/rancher/wrangler/v2/pkg/summary"
|
|
||||||
"github.com/sirupsen/logrus"
|
"github.com/sirupsen/logrus"
|
||||||
"golang.org/x/sync/errgroup"
|
"golang.org/x/sync/errgroup"
|
||||||
"k8s.io/apimachinery/pkg/api/meta"
|
"k8s.io/apimachinery/pkg/api/meta"
|
||||||
@ -34,9 +25,23 @@ import (
|
|||||||
"k8s.io/client-go/dynamic"
|
"k8s.io/client-go/dynamic"
|
||||||
"k8s.io/client-go/kubernetes"
|
"k8s.io/client-go/kubernetes"
|
||||||
"k8s.io/client-go/rest"
|
"k8s.io/client-go/rest"
|
||||||
|
|
||||||
|
"github.com/rancher/apiserver/pkg/apierror"
|
||||||
|
"github.com/rancher/apiserver/pkg/types"
|
||||||
|
"github.com/rancher/steve/pkg/accesscontrol"
|
||||||
|
"github.com/rancher/steve/pkg/attributes"
|
||||||
|
metricsStore "github.com/rancher/steve/pkg/stores/metrics"
|
||||||
|
"github.com/rancher/steve/pkg/stores/partition"
|
||||||
|
"github.com/rancher/wrangler/v2/pkg/data"
|
||||||
|
corecontrollers "github.com/rancher/wrangler/v2/pkg/generated/controllers/core/v1"
|
||||||
|
"github.com/rancher/wrangler/v2/pkg/schemas/validation"
|
||||||
|
"github.com/rancher/wrangler/v2/pkg/summary"
|
||||||
)
|
)
|
||||||
|
|
||||||
const watchTimeoutEnv = "CATTLE_WATCH_TIMEOUT_SECONDS"
|
const (
|
||||||
|
watchTimeoutEnv = "CATTLE_WATCH_TIMEOUT_SECONDS"
|
||||||
|
errNamespaceRequired = "metadata.namespace is required"
|
||||||
|
)
|
||||||
|
|
||||||
var (
|
var (
|
||||||
lowerChars = regexp.MustCompile("[a-z]+")
|
lowerChars = regexp.MustCompile("[a-z]+")
|
||||||
@ -422,20 +427,27 @@ func (s *Store) Create(apiOp *types.APIRequest, schema *types.APISchema, params
|
|||||||
}
|
}
|
||||||
|
|
||||||
name := types.Name(input)
|
name := types.Name(input)
|
||||||
ns := types.Namespace(input)
|
namespace := types.Namespace(input)
|
||||||
if name == "" && input.String("metadata", "generateName") == "" {
|
generateName := input.String("metadata", "generateName")
|
||||||
input.SetNested(schema.ID[0:1]+"-", "metadata", "generatedName")
|
|
||||||
|
if name == "" && generateName == "" {
|
||||||
|
input.SetNested(schema.ID[0:1]+"-", "metadata", "generateName")
|
||||||
}
|
}
|
||||||
if ns == "" && apiOp.Namespace != "" {
|
|
||||||
ns = apiOp.Namespace
|
if attributes.Namespaced(schema) && namespace == "" {
|
||||||
input.SetNested(ns, "metadata", "namespace")
|
if apiOp.Namespace == "" {
|
||||||
|
return nil, nil, apierror.NewAPIError(validation.InvalidBodyContent, errNamespaceRequired)
|
||||||
|
}
|
||||||
|
|
||||||
|
namespace = apiOp.Namespace
|
||||||
|
input.SetNested(namespace, "metadata", "namespace")
|
||||||
}
|
}
|
||||||
|
|
||||||
gvk := attributes.GVK(schema)
|
gvk := attributes.GVK(schema)
|
||||||
input["apiVersion"], input["kind"] = gvk.ToAPIVersionAndKind()
|
input["apiVersion"], input["kind"] = gvk.ToAPIVersionAndKind()
|
||||||
|
|
||||||
buffer := WarningBuffer{}
|
buffer := WarningBuffer{}
|
||||||
k8sClient, err := metricsStore.Wrap(s.clientGetter.TableClient(apiOp, schema, ns, &buffer))
|
k8sClient, err := metricsStore.Wrap(s.clientGetter.TableClient(apiOp, schema, namespace, &buffer))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, nil, err
|
return nil, nil, err
|
||||||
}
|
}
|
||||||
|
@ -2,18 +2,18 @@ package proxy
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"net/http"
|
"net/http"
|
||||||
|
"net/url"
|
||||||
"strings"
|
"strings"
|
||||||
"testing"
|
"testing"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/pkg/errors"
|
"github.com/pkg/errors"
|
||||||
"github.com/rancher/apiserver/pkg/types"
|
|
||||||
"github.com/rancher/steve/pkg/client"
|
|
||||||
"github.com/rancher/wrangler/v2/pkg/schemas"
|
|
||||||
"github.com/stretchr/testify/assert"
|
"github.com/stretchr/testify/assert"
|
||||||
"golang.org/x/sync/errgroup"
|
"golang.org/x/sync/errgroup"
|
||||||
v1 "k8s.io/api/core/v1"
|
v1 "k8s.io/api/core/v1"
|
||||||
|
apierrors "k8s.io/apimachinery/pkg/api/errors"
|
||||||
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
||||||
|
"k8s.io/apimachinery/pkg/apis/meta/v1/unstructured"
|
||||||
"k8s.io/apimachinery/pkg/runtime"
|
"k8s.io/apimachinery/pkg/runtime"
|
||||||
schema2 "k8s.io/apimachinery/pkg/runtime/schema"
|
schema2 "k8s.io/apimachinery/pkg/runtime/schema"
|
||||||
"k8s.io/apimachinery/pkg/util/sets"
|
"k8s.io/apimachinery/pkg/util/sets"
|
||||||
@ -22,6 +22,12 @@ import (
|
|||||||
"k8s.io/client-go/dynamic/fake"
|
"k8s.io/client-go/dynamic/fake"
|
||||||
"k8s.io/client-go/rest"
|
"k8s.io/client-go/rest"
|
||||||
clientgotesting "k8s.io/client-go/testing"
|
clientgotesting "k8s.io/client-go/testing"
|
||||||
|
|
||||||
|
"github.com/rancher/apiserver/pkg/apierror"
|
||||||
|
"github.com/rancher/apiserver/pkg/types"
|
||||||
|
"github.com/rancher/steve/pkg/client"
|
||||||
|
"github.com/rancher/wrangler/v2/pkg/schemas"
|
||||||
|
"github.com/rancher/wrangler/v2/pkg/schemas/validation"
|
||||||
)
|
)
|
||||||
|
|
||||||
var c *watch.FakeWatcher
|
var c *watch.FakeWatcher
|
||||||
@ -32,6 +38,14 @@ type testFactory struct {
|
|||||||
fakeClient *fake.FakeDynamicClient
|
fakeClient *fake.FakeDynamicClient
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (t *testFactory) TableAdminClientForWatch(ctx *types.APIRequest, schema *types.APISchema, namespace string, warningHandler rest.WarningHandler) (dynamic.ResourceInterface, error) {
|
||||||
|
return t.fakeClient.Resource(schema2.GroupVersionResource{}), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (t *testFactory) TableClient(ctx *types.APIRequest, schema *types.APISchema, namespace string, warningHandler rest.WarningHandler) (dynamic.ResourceInterface, error) {
|
||||||
|
return t.fakeClient.Resource(schema2.GroupVersionResource{}).Namespace(namespace), nil
|
||||||
|
}
|
||||||
|
|
||||||
func TestWatchNamesErrReceive(t *testing.T) {
|
func TestWatchNamesErrReceive(t *testing.T) {
|
||||||
testClientFactory, err := client.NewFactory(&rest.Config{}, false)
|
testClientFactory, err := client.NewFactory(&rest.Config{}, false)
|
||||||
assert.Nil(t, err)
|
assert.Nil(t, err)
|
||||||
@ -80,10 +94,6 @@ func TestByNames(t *testing.T) {
|
|||||||
assert.Nil(t, warn)
|
assert.Nil(t, warn)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (t *testFactory) TableAdminClientForWatch(ctx *types.APIRequest, schema *types.APISchema, namespace string, warningHandler rest.WarningHandler) (dynamic.ResourceInterface, error) {
|
|
||||||
return t.fakeClient.Resource(schema2.GroupVersionResource{}), nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func receiveUntil(wc chan watch.Event, d time.Duration) error {
|
func receiveUntil(wc chan watch.Event, d time.Duration) error {
|
||||||
timer := time.NewTicker(d)
|
timer := time.NewTicker(d)
|
||||||
defer timer.Stop()
|
defer timer.Stop()
|
||||||
@ -121,3 +131,357 @@ func receiveUntil(wc chan watch.Event, d time.Duration) error {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestCreate(t *testing.T) {
|
||||||
|
type input struct {
|
||||||
|
apiOp *types.APIRequest
|
||||||
|
schema *types.APISchema
|
||||||
|
params types.APIObject
|
||||||
|
}
|
||||||
|
|
||||||
|
type expected struct {
|
||||||
|
value *unstructured.Unstructured
|
||||||
|
warning []types.Warning
|
||||||
|
err error
|
||||||
|
}
|
||||||
|
|
||||||
|
testCases := []struct {
|
||||||
|
name string
|
||||||
|
namespace string
|
||||||
|
input input
|
||||||
|
expected expected
|
||||||
|
createReactorFunc clientgotesting.ReactionFunc
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
name: "creating resource - namespace scoped",
|
||||||
|
input: input{
|
||||||
|
apiOp: &types.APIRequest{
|
||||||
|
Schema: &types.APISchema{
|
||||||
|
Schema: &schemas.Schema{
|
||||||
|
ID: "testing",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
Request: &http.Request{URL: &url.URL{}},
|
||||||
|
},
|
||||||
|
schema: &types.APISchema{
|
||||||
|
Schema: &schemas.Schema{
|
||||||
|
ID: "testing",
|
||||||
|
Attributes: map[string]interface{}{
|
||||||
|
"version": "v1",
|
||||||
|
"kind": "Secret",
|
||||||
|
"namespaced": true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
params: types.APIObject{
|
||||||
|
Object: map[string]interface{}{
|
||||||
|
"apiVersion": "v1",
|
||||||
|
"kind": "Secret",
|
||||||
|
"metadata": map[string]interface{}{
|
||||||
|
"name": "testing-secret",
|
||||||
|
"namespace": "testing-ns",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
createReactorFunc: func(action clientgotesting.Action) (handled bool, ret runtime.Object, err error) {
|
||||||
|
return false, ret, nil
|
||||||
|
},
|
||||||
|
expected: expected{
|
||||||
|
value: &unstructured.Unstructured{Object: map[string]interface{}{
|
||||||
|
"apiVersion": "v1",
|
||||||
|
"kind": "Secret",
|
||||||
|
"metadata": map[string]interface{}{
|
||||||
|
"name": "testing-secret",
|
||||||
|
"namespace": "testing-ns",
|
||||||
|
},
|
||||||
|
}},
|
||||||
|
warning: []types.Warning{},
|
||||||
|
err: nil,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "creating resource - cluster scoped",
|
||||||
|
input: input{
|
||||||
|
apiOp: &types.APIRequest{
|
||||||
|
Schema: &types.APISchema{
|
||||||
|
Schema: &schemas.Schema{
|
||||||
|
ID: "testing",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
Request: &http.Request{URL: &url.URL{}},
|
||||||
|
},
|
||||||
|
schema: &types.APISchema{
|
||||||
|
Schema: &schemas.Schema{
|
||||||
|
ID: "testing",
|
||||||
|
Attributes: map[string]interface{}{
|
||||||
|
"version": "v1",
|
||||||
|
"kind": "Secret",
|
||||||
|
"namespaced": false,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
params: types.APIObject{
|
||||||
|
Object: map[string]interface{}{
|
||||||
|
"apiVersion": "v1",
|
||||||
|
"kind": "Secret",
|
||||||
|
"metadata": map[string]interface{}{
|
||||||
|
"name": "testing-secret",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
createReactorFunc: func(action clientgotesting.Action) (handled bool, ret runtime.Object, err error) {
|
||||||
|
return false, ret, nil
|
||||||
|
},
|
||||||
|
expected: expected{
|
||||||
|
value: &unstructured.Unstructured{Object: map[string]interface{}{
|
||||||
|
"apiVersion": "v1",
|
||||||
|
"kind": "Secret",
|
||||||
|
"metadata": map[string]interface{}{
|
||||||
|
"name": "testing-secret",
|
||||||
|
},
|
||||||
|
}},
|
||||||
|
warning: []types.Warning{},
|
||||||
|
err: nil,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "missing name",
|
||||||
|
input: input{
|
||||||
|
apiOp: &types.APIRequest{
|
||||||
|
Schema: &types.APISchema{
|
||||||
|
Schema: &schemas.Schema{
|
||||||
|
ID: "testing",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
Request: &http.Request{URL: &url.URL{}},
|
||||||
|
},
|
||||||
|
schema: &types.APISchema{
|
||||||
|
Schema: &schemas.Schema{
|
||||||
|
ID: "testing",
|
||||||
|
Attributes: map[string]interface{}{
|
||||||
|
"version": "v1",
|
||||||
|
"kind": "Secret",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
params: types.APIObject{
|
||||||
|
Object: map[string]interface{}{
|
||||||
|
"apiVersion": "v1",
|
||||||
|
"kind": "Secret",
|
||||||
|
"metadata": map[string]interface{}{
|
||||||
|
"namespace": "testing-ns",
|
||||||
|
"generateName": "testing-gen-name",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
createReactorFunc: func(action clientgotesting.Action) (handled bool, ret runtime.Object, err error) {
|
||||||
|
return false, ret, nil
|
||||||
|
},
|
||||||
|
expected: expected{
|
||||||
|
value: &unstructured.Unstructured{Object: map[string]interface{}{
|
||||||
|
"apiVersion": "v1",
|
||||||
|
"kind": "Secret",
|
||||||
|
"metadata": map[string]interface{}{
|
||||||
|
"generateName": "testing-gen-name",
|
||||||
|
"namespace": "testing-ns",
|
||||||
|
},
|
||||||
|
}},
|
||||||
|
warning: []types.Warning{},
|
||||||
|
err: nil,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "missing name / generateName",
|
||||||
|
input: input{
|
||||||
|
apiOp: &types.APIRequest{
|
||||||
|
Schema: &types.APISchema{
|
||||||
|
Schema: &schemas.Schema{
|
||||||
|
ID: "testing",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
Request: &http.Request{URL: &url.URL{}},
|
||||||
|
},
|
||||||
|
schema: &types.APISchema{
|
||||||
|
Schema: &schemas.Schema{
|
||||||
|
ID: "testing",
|
||||||
|
Attributes: map[string]interface{}{
|
||||||
|
"version": "v1",
|
||||||
|
"kind": "Secret",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
params: types.APIObject{
|
||||||
|
Object: map[string]interface{}{
|
||||||
|
"apiVersion": "v1",
|
||||||
|
"kind": "Secret",
|
||||||
|
"metadata": map[string]interface{}{
|
||||||
|
"namespace": "testing-ns",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
createReactorFunc: func(action clientgotesting.Action) (handled bool, ret runtime.Object, err error) {
|
||||||
|
return false, ret, nil
|
||||||
|
},
|
||||||
|
expected: expected{
|
||||||
|
value: &unstructured.Unstructured{Object: map[string]interface{}{
|
||||||
|
"apiVersion": "v1",
|
||||||
|
"kind": "Secret",
|
||||||
|
"metadata": map[string]interface{}{
|
||||||
|
"generateName": "t-",
|
||||||
|
"namespace": "testing-ns",
|
||||||
|
},
|
||||||
|
}},
|
||||||
|
warning: []types.Warning{},
|
||||||
|
err: nil,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "missing namespace in the params / should copy from apiOp",
|
||||||
|
input: input{
|
||||||
|
apiOp: &types.APIRequest{
|
||||||
|
Schema: &types.APISchema{
|
||||||
|
Schema: &schemas.Schema{
|
||||||
|
ID: "testing",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
Namespace: "testing-ns",
|
||||||
|
Request: &http.Request{URL: &url.URL{}},
|
||||||
|
},
|
||||||
|
schema: &types.APISchema{
|
||||||
|
Schema: &schemas.Schema{
|
||||||
|
ID: "testing",
|
||||||
|
Attributes: map[string]interface{}{
|
||||||
|
"version": "v1",
|
||||||
|
"kind": "Secret",
|
||||||
|
"namespaced": true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
params: types.APIObject{
|
||||||
|
Object: map[string]interface{}{
|
||||||
|
"apiVersion": "v1",
|
||||||
|
"kind": "Secret",
|
||||||
|
"metadata": map[string]interface{}{
|
||||||
|
"name": "testing-secret",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
createReactorFunc: func(action clientgotesting.Action) (handled bool, ret runtime.Object, err error) {
|
||||||
|
return false, ret, nil
|
||||||
|
},
|
||||||
|
expected: expected{
|
||||||
|
value: &unstructured.Unstructured{Object: map[string]interface{}{
|
||||||
|
"apiVersion": "v1",
|
||||||
|
"kind": "Secret",
|
||||||
|
"metadata": map[string]interface{}{
|
||||||
|
"name": "testing-secret",
|
||||||
|
"namespace": "testing-ns",
|
||||||
|
},
|
||||||
|
}},
|
||||||
|
warning: []types.Warning{},
|
||||||
|
err: nil,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "missing namespace - namespace scoped",
|
||||||
|
input: input{
|
||||||
|
apiOp: &types.APIRequest{
|
||||||
|
Schema: &types.APISchema{
|
||||||
|
Schema: &schemas.Schema{
|
||||||
|
ID: "testing",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
Request: &http.Request{URL: &url.URL{}},
|
||||||
|
},
|
||||||
|
schema: &types.APISchema{
|
||||||
|
Schema: &schemas.Schema{
|
||||||
|
ID: "testing",
|
||||||
|
Attributes: map[string]interface{}{
|
||||||
|
"namespaced": true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
params: types.APIObject{},
|
||||||
|
},
|
||||||
|
expected: expected{
|
||||||
|
value: nil,
|
||||||
|
warning: nil,
|
||||||
|
err: apierror.NewAPIError(
|
||||||
|
validation.InvalidBodyContent,
|
||||||
|
errNamespaceRequired,
|
||||||
|
),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "error response",
|
||||||
|
input: input{
|
||||||
|
apiOp: &types.APIRequest{
|
||||||
|
Schema: &types.APISchema{
|
||||||
|
Schema: &schemas.Schema{
|
||||||
|
ID: "testing",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
Request: &http.Request{URL: &url.URL{}},
|
||||||
|
},
|
||||||
|
schema: &types.APISchema{
|
||||||
|
Schema: &schemas.Schema{
|
||||||
|
ID: "testing",
|
||||||
|
Attributes: map[string]interface{}{
|
||||||
|
"version": "v1",
|
||||||
|
"kind": "Secret",
|
||||||
|
"namespaced": true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
params: types.APIObject{
|
||||||
|
Object: map[string]interface{}{
|
||||||
|
"apiVersion": "v1",
|
||||||
|
"kind": "Secret",
|
||||||
|
"metadata": map[string]interface{}{
|
||||||
|
"name": "testing-secret",
|
||||||
|
"namespace": "testing-ns",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
createReactorFunc: func(action clientgotesting.Action) (handled bool, ret runtime.Object, err error) {
|
||||||
|
return true, nil, apierrors.NewUnauthorized("sample reason")
|
||||||
|
},
|
||||||
|
expected: expected{
|
||||||
|
value: nil,
|
||||||
|
warning: []types.Warning{},
|
||||||
|
err: apierrors.NewUnauthorized("sample reason"),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, tt := range testCases {
|
||||||
|
t.Run(tt.name, func(t *testing.T) {
|
||||||
|
testClientFactory, err := client.NewFactory(&rest.Config{}, false)
|
||||||
|
assert.Nil(t, err)
|
||||||
|
|
||||||
|
fakeClient := fake.NewSimpleDynamicClient(runtime.NewScheme())
|
||||||
|
|
||||||
|
if tt.createReactorFunc != nil {
|
||||||
|
fakeClient.PrependReactor("create", "*", tt.createReactorFunc)
|
||||||
|
}
|
||||||
|
|
||||||
|
testStore := Store{
|
||||||
|
clientGetter: &testFactory{Factory: testClientFactory,
|
||||||
|
fakeClient: fakeClient,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
value, warning, err := testStore.Create(tt.input.apiOp, tt.input.schema, tt.input.params)
|
||||||
|
|
||||||
|
assert.Equal(t, tt.expected.value, value)
|
||||||
|
assert.Equal(t, tt.expected.warning, warning)
|
||||||
|
assert.Equal(t, tt.expected.err, err)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user