components-contrib/tests/certification/bindings/localstorage/localstorage_test.go

208 lines
6.1 KiB
Go

/*
Copyright 2022 The Dapr 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 localstorage_test
import (
"os"
"strconv"
"testing"
"time"
"github.com/stretchr/testify/assert"
"github.com/stretchr/testify/require"
"github.com/dapr/components-contrib/bindings"
bindings_localstorage "github.com/dapr/components-contrib/bindings/localstorage"
"github.com/dapr/components-contrib/tests/certification/embedded"
"github.com/dapr/components-contrib/tests/certification/flow"
"github.com/dapr/components-contrib/tests/certification/flow/sidecar"
bindings_loader "github.com/dapr/dapr/pkg/components/bindings"
"github.com/dapr/dapr/pkg/runtime"
daprsdk "github.com/dapr/go-sdk/client"
"github.com/dapr/kit/logger"
)
const (
sidecarName = "localstorage-sidecar"
bindingName = "localstorage-binding"
dir = "/tmp/dapr-cert-test/bindings.localstorage"
fileName = "test.txt"
fileData = "test data"
)
func TestLocalStorage(t *testing.T) {
invokeCreateWithConfig := func(ctx flow.Context, config map[string]string) error {
client, clientErr := daprsdk.NewClientWithPort(strconv.Itoa(runtime.DefaultDaprAPIGRPCPort))
if clientErr != nil {
panic(clientErr)
}
defer client.Close()
invokeRequest := &daprsdk.InvokeBindingRequest{
Name: bindingName,
Operation: string(bindings.CreateOperation),
Data: []byte(fileData),
Metadata: config,
}
return client.InvokeOutputBinding(ctx, invokeRequest)
}
invokeGetWithConfig := func(ctx flow.Context, config map[string]string) ([]byte, error) {
client, clientErr := daprsdk.NewClientWithPort(strconv.Itoa(runtime.DefaultDaprAPIGRPCPort))
if clientErr != nil {
panic(clientErr)
}
defer client.Close()
invokeRequest := &daprsdk.InvokeBindingRequest{
Name: bindingName,
Operation: string(bindings.GetOperation),
Metadata: config,
}
rsp, err := client.InvokeBinding(ctx, invokeRequest)
if err != nil {
return nil, err
}
return rsp.Data, err
}
invokeDeleteWithConfig := func(ctx flow.Context, config map[string]string) error {
client, clientErr := daprsdk.NewClientWithPort(strconv.Itoa(runtime.DefaultDaprAPIGRPCPort))
if clientErr != nil {
panic(clientErr)
}
defer client.Close()
invokeRequest := &daprsdk.InvokeBindingRequest{
Name: bindingName,
Operation: string(bindings.DeleteOperation),
Metadata: config,
}
return client.InvokeOutputBinding(ctx, invokeRequest)
}
invokeListWithConfig := func(ctx flow.Context, config map[string]string) ([]byte, error) {
client, clientErr := daprsdk.NewClientWithPort(strconv.Itoa(runtime.DefaultDaprAPIGRPCPort))
if clientErr != nil {
panic(clientErr)
}
defer client.Close()
invokeRequest := &daprsdk.InvokeBindingRequest{
Name: bindingName,
Operation: string(bindings.ListOperation),
Metadata: config,
}
rsp, err := client.InvokeBinding(ctx, invokeRequest)
if err != nil {
return nil, err
}
return rsp.Data, err
}
testInvokeCreateAndGet := func(ctx flow.Context) error {
// sleep to avoid metadata request rate limit before initializing new client
flow.Sleep(3 * time.Second)
err := invokeCreateWithConfig(ctx, map[string]string{"fileName": fileName})
require.NoError(t, err)
raw, err := invokeGetWithConfig(ctx, map[string]string{"fileName": fileName})
require.NoError(t, err)
assert.Equal(t, fileData, string(raw))
return clear()
}
testInvokeGetFileNotExists := func(ctx flow.Context) error {
// sleep to avoid metadata request rate limit before initializing new client
flow.Sleep(3 * time.Second)
_, err := invokeGetWithConfig(ctx, map[string]string{"fileName": "not-exists.txt"})
require.Error(t, err)
require.ErrorContains(t, err, "file not found:")
require.ErrorContains(t, err, "not-exists.txt")
return clear()
}
testInvokeDeleteFile := func(ctx flow.Context) error {
// sleep to avoid metadata request rate limit before initializing new client
flow.Sleep(3 * time.Second)
err := invokeCreateWithConfig(ctx, map[string]string{"fileName": fileName})
require.NoError(t, err)
err = invokeDeleteWithConfig(ctx, map[string]string{"fileName": fileName})
require.NoError(t, err)
return clear()
}
testInvokeListFiles := func(ctx flow.Context) error {
// sleep to avoid metadata request rate limit before initializing new client
flow.Sleep(3 * time.Second)
fileNames := []string{fileName + "1", fileName + "2", fileName + "3"}
for _, fileName := range fileNames {
err := invokeCreateWithConfig(ctx, map[string]string{"fileName": fileName})
require.NoError(t, err)
}
raw, err := invokeListWithConfig(ctx, nil)
require.NoError(t, err)
for _, fileName := range fileNames {
assert.Contains(t, string(raw), fileName)
}
// clear
return clear()
}
flow.New(t, "test local storage operations").
Step(sidecar.Run(sidecarName,
append(componentRuntimeOptions(),
embedded.WithoutApp(),
embedded.WithComponentsPath("./components"),
embedded.WithDaprGRPCPort(strconv.Itoa(runtime.DefaultDaprAPIGRPCPort)),
embedded.WithDaprHTTPPort(strconv.Itoa(runtime.DefaultDaprHTTPPort)),
)...,
)).
Step("create file and get data success", testInvokeCreateAndGet).
Step("get error when file not exists", testInvokeGetFileNotExists).
Step("delete file success", testInvokeDeleteFile).
Step("list files", testInvokeListFiles).
Run()
}
func clear() error {
return os.RemoveAll(dir)
}
func componentRuntimeOptions() []embedded.Option {
log := logger.NewLogger("dapr.components")
bindingsRegistry := bindings_loader.NewRegistry()
bindingsRegistry.Logger = log
bindingsRegistry.RegisterOutputBinding(bindings_localstorage.NewLocalStorage, "localstorage")
return []embedded.Option{
embedded.WithBindings(bindingsRegistry),
}
}