2024-07-23 01:08:30 +08:00
|
|
|
package test
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"testing"
|
2024-07-31 08:39:33 +08:00
|
|
|
"time"
|
2024-07-23 01:08:30 +08:00
|
|
|
|
2024-10-17 18:18:29 +08:00
|
|
|
"github.com/prometheus/client_golang/prometheus"
|
2024-07-23 01:08:30 +08:00
|
|
|
"github.com/stretchr/testify/require"
|
2024-07-26 00:17:39 +08:00
|
|
|
"google.golang.org/grpc"
|
|
|
|
"google.golang.org/grpc/credentials/insecure"
|
2024-07-23 01:08:30 +08:00
|
|
|
|
2024-08-12 14:26:53 +08:00
|
|
|
"github.com/grafana/authlib/claims"
|
2024-07-26 00:17:39 +08:00
|
|
|
"github.com/grafana/dskit/services"
|
2024-10-24 15:12:37 +08:00
|
|
|
|
2024-07-26 00:17:39 +08:00
|
|
|
"github.com/grafana/grafana/pkg/apimachinery/identity"
|
2024-07-23 01:08:30 +08:00
|
|
|
infraDB "github.com/grafana/grafana/pkg/infra/db"
|
2024-10-28 20:35:30 +08:00
|
|
|
"github.com/grafana/grafana/pkg/infra/tracing"
|
2024-07-23 01:08:30 +08:00
|
|
|
"github.com/grafana/grafana/pkg/services/featuremgmt"
|
|
|
|
"github.com/grafana/grafana/pkg/setting"
|
|
|
|
"github.com/grafana/grafana/pkg/storage/unified/resource"
|
|
|
|
"github.com/grafana/grafana/pkg/storage/unified/sql"
|
|
|
|
"github.com/grafana/grafana/pkg/storage/unified/sql/db/dbimpl"
|
|
|
|
"github.com/grafana/grafana/pkg/tests/testsuite"
|
|
|
|
"github.com/grafana/grafana/pkg/util/testutil"
|
|
|
|
)
|
|
|
|
|
|
|
|
func TestMain(m *testing.M) {
|
|
|
|
testsuite.Run(m)
|
|
|
|
}
|
|
|
|
|
2024-07-31 17:05:59 +08:00
|
|
|
func newServer(t *testing.T) (sql.Backend, resource.ResourceServer) {
|
2024-07-23 01:08:30 +08:00
|
|
|
t.Helper()
|
|
|
|
|
|
|
|
dbstore := infraDB.InitTestDB(t)
|
|
|
|
cfg := setting.NewCfg()
|
|
|
|
|
2024-10-04 20:07:20 +08:00
|
|
|
eDB, err := dbimpl.ProvideResourceDB(dbstore, cfg, nil)
|
2024-07-23 01:08:30 +08:00
|
|
|
require.NoError(t, err)
|
|
|
|
require.NotNil(t, eDB)
|
|
|
|
|
|
|
|
ret, err := sql.NewBackend(sql.BackendOptions{
|
|
|
|
DBProvider: eDB,
|
|
|
|
})
|
|
|
|
require.NoError(t, err)
|
|
|
|
require.NotNil(t, ret)
|
|
|
|
|
|
|
|
err = ret.Init(testutil.NewDefaultTestContext(t))
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
2024-07-31 17:05:59 +08:00
|
|
|
server, err := resource.NewResourceServer(resource.ResourceServerOptions{
|
|
|
|
Backend: ret,
|
|
|
|
Diagnostics: ret,
|
|
|
|
Lifecycle: ret,
|
|
|
|
})
|
|
|
|
require.NoError(t, err)
|
|
|
|
require.NotNil(t, server)
|
|
|
|
|
|
|
|
return ret, server
|
2024-07-23 01:08:30 +08:00
|
|
|
}
|
|
|
|
|
2024-07-30 03:49:54 +08:00
|
|
|
func TestIntegrationBackendHappyPath(t *testing.T) {
|
2024-10-11 17:11:33 +08:00
|
|
|
if infraDB.IsTestDbSQLite() {
|
|
|
|
t.Skip("TODO: test blocking, skipping to unblock Enterprise until we fix this")
|
|
|
|
}
|
2024-07-30 03:49:54 +08:00
|
|
|
if testing.Short() {
|
|
|
|
t.Skip("skipping integration test")
|
|
|
|
}
|
2024-07-24 06:08:47 +08:00
|
|
|
|
2024-07-31 17:05:59 +08:00
|
|
|
testUserA := &identity.StaticRequester{
|
2024-08-12 14:26:53 +08:00
|
|
|
Type: claims.TypeUser,
|
2024-07-31 17:05:59 +08:00
|
|
|
Login: "testuser",
|
|
|
|
UserID: 123,
|
|
|
|
UserUID: "u123",
|
|
|
|
OrgRole: identity.RoleAdmin,
|
|
|
|
IsGrafanaAdmin: true, // can do anything
|
|
|
|
}
|
|
|
|
ctx := identity.WithRequester(context.Background(), testUserA)
|
|
|
|
backend, server := newServer(t)
|
2024-07-23 01:08:30 +08:00
|
|
|
|
2024-07-31 17:05:59 +08:00
|
|
|
stream, err := backend.WatchWriteEvents(context.Background()) // Using a different context to avoid canceling the stream after the DefaultContextTimeout
|
2024-07-30 03:49:54 +08:00
|
|
|
require.NoError(t, err)
|
2024-10-11 17:11:33 +08:00
|
|
|
var rv1, rv2, rv3, rv4, rv5 int64
|
2024-07-23 01:08:30 +08:00
|
|
|
|
|
|
|
t.Run("Add 3 resources", func(t *testing.T) {
|
2024-10-11 17:11:33 +08:00
|
|
|
rv1, err = writeEvent(ctx, backend, "item1", resource.WatchEvent_ADDED)
|
2024-07-30 03:49:54 +08:00
|
|
|
require.NoError(t, err)
|
2024-10-11 17:11:33 +08:00
|
|
|
require.Greater(t, rv1, int64(0))
|
2024-07-23 01:08:30 +08:00
|
|
|
|
2024-10-11 17:11:33 +08:00
|
|
|
rv2, err = writeEvent(ctx, backend, "item2", resource.WatchEvent_ADDED)
|
2024-07-30 03:49:54 +08:00
|
|
|
require.NoError(t, err)
|
2024-10-11 17:11:33 +08:00
|
|
|
require.Greater(t, rv2, rv1)
|
2024-07-23 01:08:30 +08:00
|
|
|
|
2024-10-11 17:11:33 +08:00
|
|
|
rv3, err = writeEvent(ctx, backend, "item3", resource.WatchEvent_ADDED)
|
2024-07-30 03:49:54 +08:00
|
|
|
require.NoError(t, err)
|
2024-10-11 17:11:33 +08:00
|
|
|
require.Greater(t, rv3, rv2)
|
2024-07-23 01:08:30 +08:00
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("Update item2", func(t *testing.T) {
|
2024-10-11 17:11:33 +08:00
|
|
|
rv4, err = writeEvent(ctx, backend, "item2", resource.WatchEvent_MODIFIED)
|
2024-07-30 03:49:54 +08:00
|
|
|
require.NoError(t, err)
|
2024-10-11 17:11:33 +08:00
|
|
|
require.Greater(t, rv4, rv3)
|
2024-07-23 01:08:30 +08:00
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("Delete item1", func(t *testing.T) {
|
2024-10-11 17:11:33 +08:00
|
|
|
rv5, err = writeEvent(ctx, backend, "item1", resource.WatchEvent_DELETED)
|
2024-07-30 03:49:54 +08:00
|
|
|
require.NoError(t, err)
|
2024-10-11 17:11:33 +08:00
|
|
|
require.Greater(t, rv5, rv4)
|
2024-07-23 01:08:30 +08:00
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("Read latest item 2", func(t *testing.T) {
|
2024-07-31 17:05:59 +08:00
|
|
|
resp := backend.ReadResource(ctx, &resource.ReadRequest{Key: resourceKey("item2")})
|
|
|
|
require.Nil(t, resp.Error)
|
2024-10-11 17:11:33 +08:00
|
|
|
require.Equal(t, rv4, resp.ResourceVersion)
|
2024-07-30 03:49:54 +08:00
|
|
|
require.Equal(t, "item2 MODIFIED", string(resp.Value))
|
2024-07-23 01:08:30 +08:00
|
|
|
})
|
|
|
|
|
2024-07-31 17:05:59 +08:00
|
|
|
t.Run("Read early version of item2", func(t *testing.T) {
|
|
|
|
resp := backend.ReadResource(ctx, &resource.ReadRequest{
|
2024-07-23 01:08:30 +08:00
|
|
|
Key: resourceKey("item2"),
|
2024-10-11 17:11:33 +08:00
|
|
|
ResourceVersion: rv3, // item2 was created at rv2 and updated at rv4
|
2024-07-23 01:08:30 +08:00
|
|
|
})
|
2024-07-31 17:05:59 +08:00
|
|
|
require.Nil(t, resp.Error)
|
2024-10-11 17:11:33 +08:00
|
|
|
require.Equal(t, rv2, resp.ResourceVersion)
|
2024-07-30 03:49:54 +08:00
|
|
|
require.Equal(t, "item2 ADDED", string(resp.Value))
|
2024-07-23 01:08:30 +08:00
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("PrepareList latest", func(t *testing.T) {
|
2024-07-31 17:05:59 +08:00
|
|
|
resp, err := server.List(ctx, &resource.ListRequest{
|
2024-07-23 01:08:30 +08:00
|
|
|
Options: &resource.ListOptions{
|
|
|
|
Key: &resource.ResourceKey{
|
|
|
|
Namespace: "namespace",
|
|
|
|
Group: "group",
|
|
|
|
Resource: "resource",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
})
|
2024-07-30 03:49:54 +08:00
|
|
|
require.NoError(t, err)
|
2024-07-31 17:05:59 +08:00
|
|
|
require.Nil(t, resp.Error)
|
2024-07-30 03:49:54 +08:00
|
|
|
require.Len(t, resp.Items, 2)
|
|
|
|
require.Equal(t, "item2 MODIFIED", string(resp.Items[0].Value))
|
|
|
|
require.Equal(t, "item3 ADDED", string(resp.Items[1].Value))
|
2024-10-11 17:11:33 +08:00
|
|
|
require.Equal(t, rv5, resp.ResourceVersion)
|
2024-07-23 01:08:30 +08:00
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("Watch events", func(t *testing.T) {
|
|
|
|
event := <-stream
|
2024-07-30 03:49:54 +08:00
|
|
|
require.Equal(t, "item1", event.Key.Name)
|
2024-10-11 17:11:33 +08:00
|
|
|
require.Equal(t, rv1, event.ResourceVersion)
|
2024-07-30 03:49:54 +08:00
|
|
|
require.Equal(t, resource.WatchEvent_ADDED, event.Type)
|
2024-07-23 01:08:30 +08:00
|
|
|
event = <-stream
|
2024-07-30 03:49:54 +08:00
|
|
|
require.Equal(t, "item2", event.Key.Name)
|
2024-10-11 17:11:33 +08:00
|
|
|
require.Equal(t, rv2, event.ResourceVersion)
|
2024-07-30 03:49:54 +08:00
|
|
|
require.Equal(t, resource.WatchEvent_ADDED, event.Type)
|
2024-07-23 01:08:30 +08:00
|
|
|
|
|
|
|
event = <-stream
|
2024-07-30 03:49:54 +08:00
|
|
|
require.Equal(t, "item3", event.Key.Name)
|
2024-10-11 17:11:33 +08:00
|
|
|
require.Equal(t, rv3, event.ResourceVersion)
|
2024-07-30 03:49:54 +08:00
|
|
|
require.Equal(t, resource.WatchEvent_ADDED, event.Type)
|
2024-07-23 01:08:30 +08:00
|
|
|
|
|
|
|
event = <-stream
|
2024-07-30 03:49:54 +08:00
|
|
|
require.Equal(t, "item2", event.Key.Name)
|
2024-10-11 17:11:33 +08:00
|
|
|
require.Equal(t, rv4, event.ResourceVersion)
|
2024-07-30 03:49:54 +08:00
|
|
|
require.Equal(t, resource.WatchEvent_MODIFIED, event.Type)
|
2024-07-23 01:08:30 +08:00
|
|
|
|
|
|
|
event = <-stream
|
2024-07-30 03:49:54 +08:00
|
|
|
require.Equal(t, "item1", event.Key.Name)
|
2024-10-11 17:11:33 +08:00
|
|
|
require.Equal(t, rv5, event.ResourceVersion)
|
2024-07-30 03:49:54 +08:00
|
|
|
require.Equal(t, resource.WatchEvent_DELETED, event.Type)
|
2024-07-23 01:08:30 +08:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2024-07-30 03:49:54 +08:00
|
|
|
func TestIntegrationBackendWatchWriteEventsFromLastest(t *testing.T) {
|
2024-10-11 17:11:33 +08:00
|
|
|
if infraDB.IsTestDbSQLite() {
|
|
|
|
t.Skip("TODO: test blocking, skipping to unblock Enterprise until we fix this")
|
|
|
|
}
|
2024-07-30 03:49:54 +08:00
|
|
|
if testing.Short() {
|
|
|
|
t.Skip("skipping integration test")
|
|
|
|
}
|
|
|
|
|
2024-07-31 08:39:33 +08:00
|
|
|
ctx := testutil.NewTestContext(t, time.Now().Add(5*time.Second))
|
2024-07-31 17:05:59 +08:00
|
|
|
backend, _ := newServer(t)
|
2024-07-23 01:08:30 +08:00
|
|
|
|
|
|
|
// Create a few resources before initing the watch
|
2024-07-31 17:05:59 +08:00
|
|
|
_, err := writeEvent(ctx, backend, "item1", resource.WatchEvent_ADDED)
|
2024-07-30 03:49:54 +08:00
|
|
|
require.NoError(t, err)
|
2024-07-23 01:08:30 +08:00
|
|
|
|
|
|
|
// Start the watch
|
2024-07-31 17:05:59 +08:00
|
|
|
stream, err := backend.WatchWriteEvents(ctx)
|
2024-07-30 03:49:54 +08:00
|
|
|
require.NoError(t, err)
|
2024-07-23 01:08:30 +08:00
|
|
|
|
|
|
|
// Create one more event
|
2024-07-31 17:05:59 +08:00
|
|
|
_, err = writeEvent(ctx, backend, "item2", resource.WatchEvent_ADDED)
|
2024-07-30 03:49:54 +08:00
|
|
|
require.NoError(t, err)
|
|
|
|
require.Equal(t, "item2", (<-stream).Key.Name)
|
2024-07-23 01:08:30 +08:00
|
|
|
}
|
|
|
|
|
2024-07-31 17:05:59 +08:00
|
|
|
func TestIntegrationBackendList(t *testing.T) {
|
2024-10-11 17:11:33 +08:00
|
|
|
if infraDB.IsTestDbSQLite() {
|
|
|
|
t.Skip("TODO: test blocking, skipping to unblock Enterprise until we fix this")
|
|
|
|
}
|
2024-07-30 03:49:54 +08:00
|
|
|
if testing.Short() {
|
|
|
|
t.Skip("skipping integration test")
|
|
|
|
}
|
|
|
|
|
2024-07-31 08:39:33 +08:00
|
|
|
ctx := testutil.NewTestContext(t, time.Now().Add(5*time.Second))
|
2024-07-31 17:05:59 +08:00
|
|
|
backend, server := newServer(t)
|
|
|
|
|
|
|
|
// Create a few resources before starting the watch
|
2024-10-11 17:11:33 +08:00
|
|
|
rv1, _ := writeEvent(ctx, backend, "item1", resource.WatchEvent_ADDED)
|
|
|
|
require.Greater(t, rv1, int64(0))
|
|
|
|
rv2, _ := writeEvent(ctx, backend, "item2", resource.WatchEvent_ADDED) // rv=2 - will be modified at rv=6
|
|
|
|
require.Greater(t, rv2, rv1)
|
|
|
|
rv3, _ := writeEvent(ctx, backend, "item3", resource.WatchEvent_ADDED) // rv=3 - will be deleted at rv=7
|
|
|
|
require.Greater(t, rv3, rv2)
|
|
|
|
rv4, _ := writeEvent(ctx, backend, "item4", resource.WatchEvent_ADDED)
|
|
|
|
require.Greater(t, rv4, rv3)
|
|
|
|
rv5, _ := writeEvent(ctx, backend, "item5", resource.WatchEvent_ADDED)
|
|
|
|
require.Greater(t, rv5, rv4)
|
|
|
|
rv6, _ := writeEvent(ctx, backend, "item2", resource.WatchEvent_MODIFIED)
|
|
|
|
require.Greater(t, rv6, rv5)
|
|
|
|
rv7, _ := writeEvent(ctx, backend, "item3", resource.WatchEvent_DELETED)
|
|
|
|
require.Greater(t, rv7, rv6)
|
|
|
|
rv8, _ := writeEvent(ctx, backend, "item6", resource.WatchEvent_ADDED)
|
|
|
|
require.Greater(t, rv8, rv7)
|
|
|
|
|
2024-07-23 01:08:30 +08:00
|
|
|
t.Run("fetch all latest", func(t *testing.T) {
|
2024-07-31 17:05:59 +08:00
|
|
|
res, err := server.List(ctx, &resource.ListRequest{
|
2024-07-23 01:08:30 +08:00
|
|
|
Options: &resource.ListOptions{
|
|
|
|
Key: &resource.ResourceKey{
|
|
|
|
Group: "group",
|
|
|
|
Resource: "resource",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
})
|
2024-07-31 17:05:59 +08:00
|
|
|
require.NoError(t, err)
|
2024-07-30 18:16:16 +08:00
|
|
|
require.Nil(t, res.Error)
|
2024-07-30 03:49:54 +08:00
|
|
|
require.Len(t, res.Items, 5)
|
2024-08-01 16:27:01 +08:00
|
|
|
// should be sorted by key ASC
|
|
|
|
require.Equal(t, "item1 ADDED", string(res.Items[0].Value))
|
2024-07-30 03:49:54 +08:00
|
|
|
require.Equal(t, "item2 MODIFIED", string(res.Items[1].Value))
|
2024-08-01 16:27:01 +08:00
|
|
|
require.Equal(t, "item4 ADDED", string(res.Items[2].Value))
|
|
|
|
require.Equal(t, "item5 ADDED", string(res.Items[3].Value))
|
|
|
|
require.Equal(t, "item6 ADDED", string(res.Items[4].Value))
|
2024-07-30 03:49:54 +08:00
|
|
|
|
|
|
|
require.Empty(t, res.NextPageToken)
|
2024-07-23 01:08:30 +08:00
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("list latest first page ", func(t *testing.T) {
|
2024-07-31 17:05:59 +08:00
|
|
|
res, err := server.List(ctx, &resource.ListRequest{
|
2024-07-23 01:08:30 +08:00
|
|
|
Limit: 3,
|
|
|
|
Options: &resource.ListOptions{
|
|
|
|
Key: &resource.ResourceKey{
|
|
|
|
Group: "group",
|
|
|
|
Resource: "resource",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
})
|
2024-07-31 17:05:59 +08:00
|
|
|
require.NoError(t, err)
|
2024-07-30 18:16:16 +08:00
|
|
|
require.Nil(t, res.Error)
|
2024-07-30 03:49:54 +08:00
|
|
|
require.Len(t, res.Items, 3)
|
2024-07-23 01:08:30 +08:00
|
|
|
continueToken, err := sql.GetContinueToken(res.NextPageToken)
|
2024-07-30 03:49:54 +08:00
|
|
|
require.NoError(t, err)
|
2024-08-01 16:27:01 +08:00
|
|
|
require.Equal(t, "item1 ADDED", string(res.Items[0].Value))
|
2024-07-30 03:49:54 +08:00
|
|
|
require.Equal(t, "item2 MODIFIED", string(res.Items[1].Value))
|
2024-08-01 16:27:01 +08:00
|
|
|
require.Equal(t, "item4 ADDED", string(res.Items[2].Value))
|
2024-10-11 17:11:33 +08:00
|
|
|
require.Equal(t, rv8, continueToken.ResourceVersion)
|
2024-07-23 01:08:30 +08:00
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("list at revision", func(t *testing.T) {
|
2024-07-31 17:05:59 +08:00
|
|
|
res, err := server.List(ctx, &resource.ListRequest{
|
2024-10-11 17:11:33 +08:00
|
|
|
ResourceVersion: rv4,
|
2024-07-23 01:08:30 +08:00
|
|
|
Options: &resource.ListOptions{
|
|
|
|
Key: &resource.ResourceKey{
|
|
|
|
Group: "group",
|
|
|
|
Resource: "resource",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
})
|
2024-07-31 17:05:59 +08:00
|
|
|
require.NoError(t, err)
|
2024-07-30 18:16:16 +08:00
|
|
|
require.Nil(t, res.Error)
|
2024-07-30 03:49:54 +08:00
|
|
|
require.Len(t, res.Items, 4)
|
2024-08-01 16:27:01 +08:00
|
|
|
require.Equal(t, "item1 ADDED", string(res.Items[0].Value))
|
|
|
|
require.Equal(t, "item2 ADDED", string(res.Items[1].Value))
|
|
|
|
require.Equal(t, "item3 ADDED", string(res.Items[2].Value))
|
|
|
|
require.Equal(t, "item4 ADDED", string(res.Items[3].Value))
|
2024-07-30 03:49:54 +08:00
|
|
|
require.Empty(t, res.NextPageToken)
|
2024-07-23 01:08:30 +08:00
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("fetch first page at revision with limit", func(t *testing.T) {
|
2024-07-31 17:05:59 +08:00
|
|
|
res, err := server.List(ctx, &resource.ListRequest{
|
2024-07-23 01:08:30 +08:00
|
|
|
Limit: 3,
|
2024-10-11 17:11:33 +08:00
|
|
|
ResourceVersion: rv7,
|
2024-07-23 01:08:30 +08:00
|
|
|
Options: &resource.ListOptions{
|
|
|
|
Key: &resource.ResourceKey{
|
|
|
|
Group: "group",
|
|
|
|
Resource: "resource",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
})
|
2024-07-31 17:05:59 +08:00
|
|
|
require.NoError(t, err)
|
|
|
|
require.NoError(t, err)
|
2024-07-30 18:16:16 +08:00
|
|
|
require.Nil(t, res.Error)
|
2024-07-30 03:49:54 +08:00
|
|
|
require.Len(t, res.Items, 3)
|
|
|
|
t.Log(res.Items)
|
2024-08-01 16:27:01 +08:00
|
|
|
require.Equal(t, "item1 ADDED", string(res.Items[0].Value))
|
|
|
|
require.Equal(t, "item2 MODIFIED", string(res.Items[1].Value))
|
2024-07-30 03:49:54 +08:00
|
|
|
require.Equal(t, "item4 ADDED", string(res.Items[2].Value))
|
2024-07-23 01:08:30 +08:00
|
|
|
|
|
|
|
continueToken, err := sql.GetContinueToken(res.NextPageToken)
|
2024-07-30 03:49:54 +08:00
|
|
|
require.NoError(t, err)
|
2024-10-11 17:11:33 +08:00
|
|
|
require.Equal(t, rv7, continueToken.ResourceVersion)
|
2024-07-23 01:08:30 +08:00
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("fetch second page at revision", func(t *testing.T) {
|
|
|
|
continueToken := &sql.ContinueToken{
|
2024-10-11 17:11:33 +08:00
|
|
|
ResourceVersion: rv8,
|
2024-07-23 01:08:30 +08:00
|
|
|
StartOffset: 2,
|
|
|
|
}
|
2024-07-31 17:05:59 +08:00
|
|
|
res, err := server.List(ctx, &resource.ListRequest{
|
2024-07-23 01:08:30 +08:00
|
|
|
NextPageToken: continueToken.String(),
|
|
|
|
Limit: 2,
|
|
|
|
Options: &resource.ListOptions{
|
|
|
|
Key: &resource.ResourceKey{
|
|
|
|
Group: "group",
|
|
|
|
Resource: "resource",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
})
|
2024-07-31 17:05:59 +08:00
|
|
|
require.NoError(t, err)
|
2024-07-30 18:16:16 +08:00
|
|
|
require.Nil(t, res.Error)
|
2024-07-30 03:49:54 +08:00
|
|
|
require.Len(t, res.Items, 2)
|
2024-07-31 17:05:59 +08:00
|
|
|
t.Log(res.Items)
|
2024-08-01 16:27:01 +08:00
|
|
|
require.Equal(t, "item4 ADDED", string(res.Items[0].Value))
|
|
|
|
require.Equal(t, "item5 ADDED", string(res.Items[1].Value))
|
2024-07-23 01:08:30 +08:00
|
|
|
|
2024-07-31 17:05:59 +08:00
|
|
|
continueToken, err = sql.GetContinueToken(res.NextPageToken)
|
2024-07-30 03:49:54 +08:00
|
|
|
require.NoError(t, err)
|
2024-10-11 17:11:33 +08:00
|
|
|
require.Equal(t, rv8, continueToken.ResourceVersion)
|
2024-07-30 03:49:54 +08:00
|
|
|
require.Equal(t, int64(4), continueToken.StartOffset)
|
2024-07-23 01:08:30 +08:00
|
|
|
})
|
|
|
|
}
|
2024-07-26 00:17:39 +08:00
|
|
|
func TestClientServer(t *testing.T) {
|
2024-10-11 17:11:33 +08:00
|
|
|
if infraDB.IsTestDbSQLite() {
|
|
|
|
t.Skip("TODO: test blocking, skipping to unblock Enterprise until we fix this")
|
|
|
|
}
|
2024-07-31 08:39:33 +08:00
|
|
|
ctx := testutil.NewTestContext(t, time.Now().Add(5*time.Second))
|
2024-07-26 00:17:39 +08:00
|
|
|
dbstore := infraDB.InitTestDB(t)
|
|
|
|
|
|
|
|
cfg := setting.NewCfg()
|
2024-09-11 16:50:14 +08:00
|
|
|
cfg.GRPCServerAddress = "localhost:0" // get a free address
|
2024-07-26 00:17:39 +08:00
|
|
|
cfg.GRPCServerNetwork = "tcp"
|
|
|
|
|
2024-08-22 00:28:30 +08:00
|
|
|
features := featuremgmt.WithFeatures()
|
2024-07-26 00:17:39 +08:00
|
|
|
|
2024-10-17 18:18:29 +08:00
|
|
|
svc, err := sql.ProvideUnifiedStorageGrpcService(cfg, features, dbstore, nil, prometheus.NewPedanticRegistry())
|
2024-07-26 00:17:39 +08:00
|
|
|
require.NoError(t, err)
|
|
|
|
var client resource.ResourceStoreClient
|
|
|
|
|
|
|
|
// Test with an admin identity
|
2024-07-31 08:39:33 +08:00
|
|
|
clientCtx := identity.WithRequester(ctx, &identity.StaticRequester{
|
2024-08-12 14:26:53 +08:00
|
|
|
Type: claims.TypeUser,
|
2024-07-26 00:17:39 +08:00
|
|
|
Login: "testuser",
|
|
|
|
UserID: 123,
|
|
|
|
UserUID: "u123",
|
|
|
|
OrgRole: identity.RoleAdmin,
|
|
|
|
IsGrafanaAdmin: true, // can do anything
|
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("Start and stop service", func(t *testing.T) {
|
|
|
|
err = services.StartAndAwaitRunning(ctx, svc)
|
|
|
|
require.NoError(t, err)
|
|
|
|
require.NotEmpty(t, svc.GetAddress())
|
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("Create a client", func(t *testing.T) {
|
|
|
|
conn, err := grpc.NewClient(svc.GetAddress(), grpc.WithTransportCredentials(insecure.NewCredentials()))
|
|
|
|
require.NoError(t, err)
|
2024-10-28 20:35:30 +08:00
|
|
|
client, err = resource.NewGRPCResourceClient(tracing.NewNoopTracerService(), conn)
|
2024-10-24 15:12:37 +08:00
|
|
|
require.NoError(t, err)
|
2024-07-26 00:17:39 +08:00
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("Create a resource", func(t *testing.T) {
|
|
|
|
raw := []byte(`{
|
|
|
|
"apiVersion": "group/v0alpha1",
|
|
|
|
"kind": "resource",
|
|
|
|
"metadata": {
|
|
|
|
"name": "item1",
|
|
|
|
"namespace": "namespace"
|
|
|
|
},
|
|
|
|
"spec": {}
|
|
|
|
}`)
|
|
|
|
resp, err := client.Create(clientCtx, &resource.CreateRequest{
|
|
|
|
Key: resourceKey("item1"),
|
|
|
|
Value: raw,
|
|
|
|
})
|
|
|
|
require.NoError(t, err)
|
|
|
|
require.Empty(t, resp.Error)
|
|
|
|
require.Greater(t, resp.ResourceVersion, int64(0))
|
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("Read the resource", func(t *testing.T) {
|
|
|
|
resp, err := client.Read(clientCtx, &resource.ReadRequest{
|
|
|
|
Key: resourceKey("item1"),
|
|
|
|
})
|
|
|
|
require.NoError(t, err)
|
|
|
|
require.Empty(t, resp.Error)
|
|
|
|
require.Greater(t, resp.ResourceVersion, int64(0))
|
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("Stop the service", func(t *testing.T) {
|
|
|
|
err = services.StopAndAwaitTerminated(ctx, svc)
|
|
|
|
require.NoError(t, err)
|
|
|
|
})
|
|
|
|
}
|
2024-07-23 01:08:30 +08:00
|
|
|
|
|
|
|
func writeEvent(ctx context.Context, store sql.Backend, name string, action resource.WatchEvent_Type) (int64, error) {
|
|
|
|
return store.WriteEvent(ctx, resource.WriteEvent{
|
|
|
|
Type: action,
|
|
|
|
Value: []byte(name + " " + resource.WatchEvent_Type_name[int32(action)]),
|
|
|
|
Key: &resource.ResourceKey{
|
|
|
|
Namespace: "namespace",
|
|
|
|
Group: "group",
|
|
|
|
Resource: "resource",
|
|
|
|
Name: name,
|
|
|
|
},
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func resourceKey(name string) *resource.ResourceKey {
|
|
|
|
return &resource.ResourceKey{
|
|
|
|
Namespace: "namespace",
|
|
|
|
Group: "group",
|
|
|
|
Resource: "resource",
|
|
|
|
Name: name,
|
|
|
|
}
|
|
|
|
}
|