mirror of
https://github.com/grafana/grafana.git
synced 2025-02-25 18:55:37 -06:00
* WIP implement generic compare interface * Use global compare fn for all entities * Lint * Update pkg/apiserver/rest/dualwriter.go Co-authored-by: Dan Cech <dcech@grafana.com> * Don't need to hash, just compare bytes * Fix tests --------- Co-authored-by: Dan Cech <dcech@grafana.com>
210 lines
6.9 KiB
Go
210 lines
6.9 KiB
Go
package scope
|
|
|
|
import (
|
|
"fmt"
|
|
"time"
|
|
|
|
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
|
"k8s.io/apimachinery/pkg/fields"
|
|
"k8s.io/apimachinery/pkg/labels"
|
|
"k8s.io/apimachinery/pkg/runtime"
|
|
"k8s.io/apiserver/pkg/registry/generic"
|
|
genericregistry "k8s.io/apiserver/pkg/registry/generic/registry"
|
|
apistore "k8s.io/apiserver/pkg/storage"
|
|
|
|
scope "github.com/grafana/grafana/pkg/apis/scope/v0alpha1"
|
|
grafanaregistry "github.com/grafana/grafana/pkg/apiserver/registry/generic"
|
|
grafanarest "github.com/grafana/grafana/pkg/apiserver/rest"
|
|
gapiutil "github.com/grafana/grafana/pkg/services/apiserver/utils"
|
|
)
|
|
|
|
var _ grafanarest.Storage = (*storage)(nil)
|
|
|
|
type storage struct {
|
|
*genericregistry.Store
|
|
}
|
|
|
|
func newScopeStorage(scheme *runtime.Scheme, optsGetter generic.RESTOptionsGetter) (*storage, error) {
|
|
strategy := grafanaregistry.NewStrategy(scheme)
|
|
|
|
resourceInfo := scope.ScopeResourceInfo
|
|
store := &genericregistry.Store{
|
|
NewFunc: resourceInfo.NewFunc,
|
|
NewListFunc: resourceInfo.NewListFunc,
|
|
KeyRootFunc: grafanaregistry.KeyRootFunc(resourceInfo.GroupResource()),
|
|
KeyFunc: grafanaregistry.NamespaceKeyFunc(resourceInfo.GroupResource()),
|
|
PredicateFunc: Matcher,
|
|
DefaultQualifiedResource: resourceInfo.GroupResource(),
|
|
SingularQualifiedResource: resourceInfo.SingularGroupResource(),
|
|
TableConvertor: gapiutil.NewTableConverter(
|
|
resourceInfo.GroupResource(),
|
|
[]metav1.TableColumnDefinition{
|
|
{Name: "Name", Type: "string", Format: "name"},
|
|
{Name: "Created At", Type: "date"},
|
|
{Name: "Title", Type: "string"},
|
|
{Name: "Filters", Type: "array"},
|
|
},
|
|
func(obj any) ([]interface{}, error) {
|
|
m, ok := obj.(*scope.Scope)
|
|
if !ok {
|
|
return nil, fmt.Errorf("expected scope")
|
|
}
|
|
return []interface{}{
|
|
m.Name,
|
|
m.CreationTimestamp.UTC().Format(time.RFC3339),
|
|
m.Spec.Title,
|
|
m.Spec.Filters,
|
|
}, nil
|
|
},
|
|
),
|
|
CreateStrategy: strategy,
|
|
UpdateStrategy: strategy,
|
|
DeleteStrategy: strategy,
|
|
}
|
|
options := &generic.StoreOptions{RESTOptions: optsGetter, AttrFunc: GetAttrs}
|
|
if err := store.CompleteWithOptions(options); err != nil {
|
|
return nil, err
|
|
}
|
|
return &storage{Store: store}, nil
|
|
}
|
|
|
|
func newScopeDashboardBindingStorage(scheme *runtime.Scheme, optsGetter generic.RESTOptionsGetter) (*storage, error) {
|
|
strategy := grafanaregistry.NewStrategy(scheme)
|
|
|
|
resourceInfo := scope.ScopeDashboardBindingResourceInfo
|
|
store := &genericregistry.Store{
|
|
NewFunc: resourceInfo.NewFunc,
|
|
NewListFunc: resourceInfo.NewListFunc,
|
|
KeyRootFunc: grafanaregistry.KeyRootFunc(resourceInfo.GroupResource()),
|
|
KeyFunc: grafanaregistry.NamespaceKeyFunc(resourceInfo.GroupResource()),
|
|
PredicateFunc: Matcher,
|
|
DefaultQualifiedResource: resourceInfo.GroupResource(),
|
|
SingularQualifiedResource: resourceInfo.SingularGroupResource(),
|
|
TableConvertor: gapiutil.NewTableConverter(
|
|
resourceInfo.GroupResource(),
|
|
[]metav1.TableColumnDefinition{
|
|
{Name: "Name", Type: "string", Format: "name"},
|
|
{Name: "Created At", Type: "date"},
|
|
{Name: "Dashboard", Type: "string"},
|
|
{Name: "Scope", Type: "string"},
|
|
},
|
|
func(obj any) ([]interface{}, error) {
|
|
m, ok := obj.(*scope.ScopeDashboardBinding)
|
|
if !ok {
|
|
return nil, fmt.Errorf("expected scope dashboard binding")
|
|
}
|
|
return []interface{}{
|
|
m.Name,
|
|
m.CreationTimestamp.UTC().Format(time.RFC3339),
|
|
m.Spec.Dashboard,
|
|
m.Spec.Scope,
|
|
}, nil
|
|
},
|
|
),
|
|
CreateStrategy: strategy,
|
|
UpdateStrategy: strategy,
|
|
DeleteStrategy: strategy,
|
|
}
|
|
options := &generic.StoreOptions{RESTOptions: optsGetter, AttrFunc: GetAttrs}
|
|
if err := store.CompleteWithOptions(options); err != nil {
|
|
return nil, err
|
|
}
|
|
return &storage{Store: store}, nil
|
|
}
|
|
|
|
func newScopeNodeStorage(scheme *runtime.Scheme, optsGetter generic.RESTOptionsGetter) (*storage, error) {
|
|
strategy := grafanaregistry.NewStrategy(scheme)
|
|
|
|
resourceInfo := scope.ScopeNodeResourceInfo
|
|
store := &genericregistry.Store{
|
|
NewFunc: resourceInfo.NewFunc,
|
|
NewListFunc: resourceInfo.NewListFunc,
|
|
KeyRootFunc: grafanaregistry.KeyRootFunc(resourceInfo.GroupResource()),
|
|
KeyFunc: grafanaregistry.NamespaceKeyFunc(resourceInfo.GroupResource()),
|
|
PredicateFunc: Matcher,
|
|
DefaultQualifiedResource: resourceInfo.GroupResource(),
|
|
SingularQualifiedResource: resourceInfo.SingularGroupResource(),
|
|
TableConvertor: gapiutil.NewTableConverter(
|
|
resourceInfo.GroupResource(),
|
|
[]metav1.TableColumnDefinition{
|
|
{Name: "Name", Type: "string", Format: "name"},
|
|
{Name: "Created At", Type: "date"},
|
|
{Name: "Title", Type: "string"},
|
|
{Name: "Parent Name", Type: "string"},
|
|
{Name: "Node Type", Type: "string"},
|
|
{Name: "Link Type", Type: "string"},
|
|
{Name: "Link ID", Type: "string"},
|
|
},
|
|
func(obj any) ([]interface{}, error) {
|
|
m, ok := obj.(*scope.ScopeNode)
|
|
if !ok {
|
|
return nil, fmt.Errorf("expected scope node")
|
|
}
|
|
return []interface{}{
|
|
m.Name,
|
|
m.CreationTimestamp.UTC().Format(time.RFC3339),
|
|
m.Spec.Title,
|
|
m.Spec.ParentName,
|
|
m.Spec.NodeType,
|
|
m.Spec.LinkType,
|
|
m.Spec.LinkID,
|
|
}, nil
|
|
},
|
|
),
|
|
CreateStrategy: strategy,
|
|
UpdateStrategy: strategy,
|
|
DeleteStrategy: strategy,
|
|
}
|
|
options := &generic.StoreOptions{RESTOptions: optsGetter, AttrFunc: GetAttrs}
|
|
if err := store.CompleteWithOptions(options); err != nil {
|
|
return nil, err
|
|
}
|
|
return &storage{Store: store}, nil
|
|
}
|
|
|
|
func GetAttrs(obj runtime.Object) (labels.Set, fields.Set, error) {
|
|
if s, ok := obj.(*scope.Scope); ok {
|
|
return labels.Set(s.Labels), SelectableScopeFields(s), nil
|
|
}
|
|
if s, ok := obj.(*scope.ScopeDashboardBinding); ok {
|
|
return labels.Set(s.Labels), SelectableScopeDashboardBindingFields(s), nil
|
|
}
|
|
if s, ok := obj.(*scope.ScopeNode); ok {
|
|
return labels.Set(s.Labels), SelectableScopeNodeFields(s), nil
|
|
}
|
|
return nil, nil, fmt.Errorf("not a scope or ScopeDashboardBinding object")
|
|
}
|
|
|
|
// Matcher returns a generic.SelectionPredicate that matches on label and field selectors.
|
|
func Matcher(label labels.Selector, field fields.Selector) apistore.SelectionPredicate {
|
|
return apistore.SelectionPredicate{
|
|
Label: label,
|
|
Field: field,
|
|
GetAttrs: GetAttrs,
|
|
}
|
|
}
|
|
|
|
func SelectableScopeFields(obj *scope.Scope) fields.Set {
|
|
return generic.MergeFieldsSets(generic.ObjectMetaFieldsSet(&obj.ObjectMeta, false), fields.Set{
|
|
"spec.title": obj.Spec.Title,
|
|
})
|
|
}
|
|
|
|
func SelectableScopeDashboardBindingFields(obj *scope.ScopeDashboardBinding) fields.Set {
|
|
return generic.MergeFieldsSets(generic.ObjectMetaFieldsSet(&obj.ObjectMeta, false), fields.Set{
|
|
"spec.scope": obj.Spec.Scope,
|
|
})
|
|
}
|
|
|
|
func SelectableScopeNodeFields(obj *scope.ScopeNode) fields.Set {
|
|
parentName := ""
|
|
|
|
if obj != nil {
|
|
parentName = obj.Spec.ParentName
|
|
}
|
|
|
|
return generic.MergeFieldsSets(generic.ObjectMetaFieldsSet(&obj.ObjectMeta, false), fields.Set{
|
|
"spec.parentName": parentName,
|
|
})
|
|
}
|