mirror of
https://github.com/grafana/grafana.git
synced 2025-02-25 18:55:37 -06:00
149 lines
3.7 KiB
Go
149 lines
3.7 KiB
Go
package client
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
|
|
"github.com/grafana/authlib/authz"
|
|
authzv1 "github.com/grafana/authlib/authz/proto/v1"
|
|
"github.com/grafana/authlib/claims"
|
|
openfgav1 "github.com/openfga/api/proto/openfga/v1"
|
|
"go.opentelemetry.io/otel"
|
|
"google.golang.org/grpc"
|
|
|
|
"github.com/grafana/grafana/pkg/apimachinery/utils"
|
|
"github.com/grafana/grafana/pkg/infra/log"
|
|
authzextv1 "github.com/grafana/grafana/pkg/services/authz/proto/v1"
|
|
"github.com/grafana/grafana/pkg/setting"
|
|
)
|
|
|
|
var _ authz.AccessClient = (*Client)(nil)
|
|
|
|
var tracer = otel.Tracer("github.com/grafana/grafana/pkg/services/authz/zanzana/client")
|
|
|
|
type ClientOption func(c *Client)
|
|
|
|
func WithTenantID(tenantID string) ClientOption {
|
|
return func(c *Client) {
|
|
c.tenantID = tenantID
|
|
}
|
|
}
|
|
|
|
func WithLogger(logger log.Logger) ClientOption {
|
|
return func(c *Client) {
|
|
c.logger = logger
|
|
}
|
|
}
|
|
|
|
type Client struct {
|
|
logger log.Logger
|
|
openfga openfgav1.OpenFGAServiceClient
|
|
authz authzv1.AuthzServiceClient
|
|
authzext authzextv1.AuthzExtentionServiceClient
|
|
tenantID string
|
|
}
|
|
|
|
func NewClient(ctx context.Context, cc grpc.ClientConnInterface, cfg *setting.Cfg) (*Client, error) {
|
|
stackID := cfg.StackID
|
|
if stackID == "" {
|
|
stackID = "default"
|
|
}
|
|
|
|
return New(
|
|
ctx,
|
|
cc,
|
|
WithTenantID(fmt.Sprintf("stacks-%s", stackID)),
|
|
WithLogger(log.New("zanzana-client")),
|
|
)
|
|
}
|
|
|
|
func New(ctx context.Context, cc grpc.ClientConnInterface, opts ...ClientOption) (*Client, error) {
|
|
c := &Client{
|
|
openfga: openfgav1.NewOpenFGAServiceClient(cc),
|
|
authz: authzv1.NewAuthzServiceClient(cc),
|
|
authzext: authzextv1.NewAuthzExtentionServiceClient(cc),
|
|
}
|
|
|
|
for _, o := range opts {
|
|
o(c)
|
|
}
|
|
|
|
if c.logger == nil {
|
|
c.logger = log.NewNopLogger()
|
|
}
|
|
|
|
return c, nil
|
|
}
|
|
|
|
// Check implements authz.AccessClient.
|
|
func (c *Client) Check(ctx context.Context, id claims.AuthInfo, req authz.CheckRequest) (authz.CheckResponse, error) {
|
|
ctx, span := tracer.Start(ctx, "authz.zanzana.client.Check")
|
|
defer span.End()
|
|
|
|
res, err := c.authz.Check(ctx, &authzv1.CheckRequest{
|
|
Subject: id.GetUID(),
|
|
Verb: req.Verb,
|
|
Group: req.Group,
|
|
Resource: req.Resource,
|
|
Namespace: req.Namespace,
|
|
Name: req.Name,
|
|
Subresource: req.Subresource,
|
|
Path: req.Path,
|
|
Folder: req.Folder,
|
|
})
|
|
|
|
if err != nil {
|
|
return authz.CheckResponse{}, err
|
|
}
|
|
|
|
return authz.CheckResponse{Allowed: res.GetAllowed()}, nil
|
|
}
|
|
|
|
func (c *Client) Compile(ctx context.Context, id claims.AuthInfo, req authz.ListRequest) (authz.ItemChecker, error) {
|
|
ctx, span := tracer.Start(ctx, "authz.zanzana.client.Compile")
|
|
defer span.End()
|
|
|
|
_, err := c.authzext.List(ctx, &authzextv1.ListRequest{
|
|
Subject: id.GetUID(),
|
|
Group: req.Group,
|
|
Verb: utils.VerbList,
|
|
Resource: req.Resource,
|
|
Namespace: req.Namespace,
|
|
})
|
|
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// FIXME: implement checker
|
|
return func(namespace, name, folder string) bool { return false }, nil
|
|
}
|
|
|
|
func (c *Client) List(ctx context.Context, id claims.AuthInfo, req authz.ListRequest) (*authzextv1.ListResponse, error) {
|
|
ctx, span := tracer.Start(ctx, "authz.zanzana.client.List")
|
|
defer span.End()
|
|
|
|
return c.authzext.List(ctx, &authzextv1.ListRequest{
|
|
Subject: id.GetUID(),
|
|
Group: req.Group,
|
|
Verb: utils.VerbList,
|
|
Resource: req.Resource,
|
|
Namespace: req.Namespace,
|
|
})
|
|
}
|
|
|
|
func (c *Client) Read(ctx context.Context, req *authzextv1.ReadRequest) (*authzextv1.ReadResponse, error) {
|
|
ctx, span := tracer.Start(ctx, "authz.zanzana.client.Read")
|
|
defer span.End()
|
|
|
|
return c.authzext.Read(ctx, req)
|
|
}
|
|
|
|
func (c *Client) Write(ctx context.Context, req *authzextv1.WriteRequest) error {
|
|
ctx, span := tracer.Start(ctx, "authz.zanzana.client.Write")
|
|
defer span.End()
|
|
|
|
_, err := c.authzext.Write(ctx, req)
|
|
return err
|
|
}
|