mirror of
https://github.com/grafana/grafana.git
synced 2025-07-30 04:42:35 +08:00

* Zanzana: Support subresources for folders * refactor * fix subresource requests * implement listing for folders subresources * teams subresources PoC * re-enable tests * use team resource def from iam * fix tests * remove unused code * refactor: rename to subresource * split resource schema * update workspaces * rename folder relation to subresource * refactor: rename folder resources to subresources * update readme * fix listing * rename params in subresource filter
122 lines
3.4 KiB
Go
122 lines
3.4 KiB
Go
package dualwrite
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
|
|
openfgav1 "github.com/openfga/api/proto/openfga/v1"
|
|
|
|
claims "github.com/grafana/authlib/types"
|
|
|
|
authzextv1 "github.com/grafana/grafana/pkg/services/authz/proto/v1"
|
|
"github.com/grafana/grafana/pkg/services/authz/zanzana"
|
|
)
|
|
|
|
// legacyTupleCollector collects tuples groupd by object and tupleKey
|
|
type legacyTupleCollector func(ctx context.Context, orgID int64) (map[string]map[string]*openfgav1.TupleKey, error)
|
|
|
|
// zanzanaTupleCollector collects tuples from zanzana for given object
|
|
type zanzanaTupleCollector func(ctx context.Context, client zanzana.Client, object string, namespace string) (map[string]*openfgav1.TupleKey, error)
|
|
|
|
type resourceReconciler struct {
|
|
name string
|
|
legacy legacyTupleCollector
|
|
zanzana zanzanaTupleCollector
|
|
client zanzana.Client
|
|
}
|
|
|
|
func newResourceReconciler(name string, legacy legacyTupleCollector, zanzana zanzanaTupleCollector, client zanzana.Client) resourceReconciler {
|
|
return resourceReconciler{name, legacy, zanzana, client}
|
|
}
|
|
|
|
func (r resourceReconciler) reconcile(ctx context.Context, namespace string) error {
|
|
info, err := claims.ParseNamespace(namespace)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// 1. Fetch grafana resources stored in grafana db.
|
|
res, err := r.legacy(ctx, info.OrgID)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to collect legacy tuples for %s: %w", r.name, err)
|
|
}
|
|
|
|
var (
|
|
writes = []*openfgav1.TupleKey{}
|
|
deletes = []*openfgav1.TupleKeyWithoutCondition{}
|
|
)
|
|
|
|
for object, tuples := range res {
|
|
// 2. Fetch all tuples for given object.
|
|
// Due to limitations in open fga api we need to collect tuples per object
|
|
zanzanaTuples, err := r.zanzana(ctx, r.client, object, namespace)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to collect zanzanaa tuples for %s: %w", r.name, err)
|
|
}
|
|
|
|
// 3. Check if tuples from grafana db exists in zanzana and if not add them to writes
|
|
for key, t := range tuples {
|
|
stored, ok := zanzanaTuples[key]
|
|
if !ok {
|
|
writes = append(writes, t)
|
|
continue
|
|
}
|
|
|
|
// 4. For folder resource tuples we also need to compare the stored subresources
|
|
if zanzana.IsFolderResourceTuple(t) && t.String() != stored.String() {
|
|
deletes = append(deletes, &openfgav1.TupleKeyWithoutCondition{
|
|
User: t.User,
|
|
Relation: t.Relation,
|
|
Object: t.Object,
|
|
})
|
|
|
|
writes = append(writes, t)
|
|
}
|
|
}
|
|
|
|
// 5. Check if tuple from zanzana don't exists in grafana db, if not add them to deletes.
|
|
for key, tuple := range zanzanaTuples {
|
|
_, ok := tuples[key]
|
|
if !ok {
|
|
deletes = append(deletes, &openfgav1.TupleKeyWithoutCondition{
|
|
User: tuple.User,
|
|
Relation: tuple.Relation,
|
|
Object: tuple.Object,
|
|
})
|
|
}
|
|
}
|
|
}
|
|
|
|
if len(writes) == 0 && len(deletes) == 0 {
|
|
return nil
|
|
}
|
|
|
|
if len(deletes) > 0 {
|
|
err := batch(deletes, 100, func(items []*openfgav1.TupleKeyWithoutCondition) error {
|
|
return r.client.Write(ctx, &authzextv1.WriteRequest{
|
|
Namespace: namespace,
|
|
Deletes: &authzextv1.WriteRequestDeletes{TupleKeys: zanzana.ToAuthzExtTupleKeysWithoutCondition(items)},
|
|
})
|
|
})
|
|
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
if len(writes) > 0 {
|
|
err := batch(writes, 100, func(items []*openfgav1.TupleKey) error {
|
|
return r.client.Write(ctx, &authzextv1.WriteRequest{
|
|
Namespace: namespace,
|
|
Writes: &authzextv1.WriteRequestWrites{TupleKeys: zanzana.ToAuthzExtTupleKeys(items)},
|
|
})
|
|
})
|
|
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|