You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
153 lines
4.7 KiB
153 lines
4.7 KiB
package integration
|
|
|
|
import (
|
|
"context"
|
|
"encoding/json"
|
|
"time"
|
|
|
|
"github.com/seaweedfs/seaweedfs/weed/glog"
|
|
"github.com/seaweedfs/seaweedfs/weed/iam/policy"
|
|
"github.com/seaweedfs/seaweedfs/weed/iam/util"
|
|
)
|
|
|
|
// RoleStoreAdapter adapts RoleStore interface to CacheableStore[*RoleDefinition]
|
|
type RoleStoreAdapter struct {
|
|
store RoleStore
|
|
}
|
|
|
|
// NewRoleStoreAdapter creates a new adapter for RoleStore
|
|
func NewRoleStoreAdapter(store RoleStore) *RoleStoreAdapter {
|
|
return &RoleStoreAdapter{store: store}
|
|
}
|
|
|
|
// Get implements CacheableStore interface
|
|
func (a *RoleStoreAdapter) Get(ctx context.Context, filerAddress string, key string) (*RoleDefinition, error) {
|
|
return a.store.GetRole(ctx, filerAddress, key)
|
|
}
|
|
|
|
// Store implements CacheableStore interface
|
|
func (a *RoleStoreAdapter) Store(ctx context.Context, filerAddress string, key string, value *RoleDefinition) error {
|
|
return a.store.StoreRole(ctx, filerAddress, key, value)
|
|
}
|
|
|
|
// Delete implements CacheableStore interface
|
|
func (a *RoleStoreAdapter) Delete(ctx context.Context, filerAddress string, key string) error {
|
|
return a.store.DeleteRole(ctx, filerAddress, key)
|
|
}
|
|
|
|
// List implements CacheableStore interface
|
|
func (a *RoleStoreAdapter) List(ctx context.Context, filerAddress string) ([]string, error) {
|
|
return a.store.ListRoles(ctx, filerAddress)
|
|
}
|
|
|
|
// GenericCachedRoleStore implements RoleStore using the generic cache
|
|
type GenericCachedRoleStore struct {
|
|
*util.CachedStore[*RoleDefinition]
|
|
adapter *RoleStoreAdapter
|
|
}
|
|
|
|
// NewGenericCachedRoleStore creates a new cached role store using generics
|
|
func NewGenericCachedRoleStore(config map[string]interface{}, filerAddressProvider func() string) (*GenericCachedRoleStore, error) {
|
|
// Create underlying filer store
|
|
filerStore, err := NewFilerRoleStore(config, filerAddressProvider)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// Parse cache configuration with defaults
|
|
cacheTTL := 5 * time.Minute
|
|
listTTL := 1 * time.Minute
|
|
maxCacheSize := int64(1000)
|
|
|
|
if config != nil {
|
|
if ttlStr, ok := config["ttl"].(string); ok && ttlStr != "" {
|
|
if parsed, err := time.ParseDuration(ttlStr); err == nil {
|
|
cacheTTL = parsed
|
|
}
|
|
}
|
|
if listTTLStr, ok := config["listTtl"].(string); ok && listTTLStr != "" {
|
|
if parsed, err := time.ParseDuration(listTTLStr); err == nil {
|
|
listTTL = parsed
|
|
}
|
|
}
|
|
if maxSize, ok := config["maxCacheSize"].(int); ok && maxSize > 0 {
|
|
maxCacheSize = int64(maxSize)
|
|
}
|
|
}
|
|
|
|
// Create adapter and generic cached store
|
|
adapter := NewRoleStoreAdapter(filerStore)
|
|
cachedStore := util.NewCachedStore(
|
|
adapter,
|
|
genericCopyRoleDefinition, // Copy function
|
|
util.CachedStoreConfig{
|
|
TTL: cacheTTL,
|
|
ListTTL: listTTL,
|
|
MaxCacheSize: maxCacheSize,
|
|
},
|
|
)
|
|
|
|
glog.V(2).Infof("Initialized GenericCachedRoleStore with TTL %v, List TTL %v, Max Cache Size %d",
|
|
cacheTTL, listTTL, maxCacheSize)
|
|
|
|
return &GenericCachedRoleStore{
|
|
CachedStore: cachedStore,
|
|
adapter: adapter,
|
|
}, nil
|
|
}
|
|
|
|
// StoreRole implements RoleStore interface
|
|
func (c *GenericCachedRoleStore) StoreRole(ctx context.Context, filerAddress string, roleName string, role *RoleDefinition) error {
|
|
return c.Store(ctx, filerAddress, roleName, role)
|
|
}
|
|
|
|
// GetRole implements RoleStore interface
|
|
func (c *GenericCachedRoleStore) GetRole(ctx context.Context, filerAddress string, roleName string) (*RoleDefinition, error) {
|
|
return c.Get(ctx, filerAddress, roleName)
|
|
}
|
|
|
|
// ListRoles implements RoleStore interface
|
|
func (c *GenericCachedRoleStore) ListRoles(ctx context.Context, filerAddress string) ([]string, error) {
|
|
return c.List(ctx, filerAddress)
|
|
}
|
|
|
|
// DeleteRole implements RoleStore interface
|
|
func (c *GenericCachedRoleStore) DeleteRole(ctx context.Context, filerAddress string, roleName string) error {
|
|
return c.Delete(ctx, filerAddress, roleName)
|
|
}
|
|
|
|
// genericCopyRoleDefinition creates a deep copy of a RoleDefinition for the generic cache
|
|
func genericCopyRoleDefinition(role *RoleDefinition) *RoleDefinition {
|
|
if role == nil {
|
|
return nil
|
|
}
|
|
|
|
result := &RoleDefinition{
|
|
RoleName: role.RoleName,
|
|
RoleArn: role.RoleArn,
|
|
Description: role.Description,
|
|
}
|
|
|
|
// Deep copy trust policy if it exists
|
|
if role.TrustPolicy != nil {
|
|
trustPolicyData, err := json.Marshal(role.TrustPolicy)
|
|
if err != nil {
|
|
glog.Errorf("Failed to marshal trust policy for deep copy: %v", err)
|
|
return nil
|
|
}
|
|
var trustPolicyCopy policy.PolicyDocument
|
|
if err := json.Unmarshal(trustPolicyData, &trustPolicyCopy); err != nil {
|
|
glog.Errorf("Failed to unmarshal trust policy for deep copy: %v", err)
|
|
return nil
|
|
}
|
|
result.TrustPolicy = &trustPolicyCopy
|
|
}
|
|
|
|
// Deep copy attached policies slice
|
|
if role.AttachedPolicies != nil {
|
|
result.AttachedPolicies = make([]string, len(role.AttachedPolicies))
|
|
copy(result.AttachedPolicies, role.AttachedPolicies)
|
|
}
|
|
|
|
return result
|
|
}
|