|
|
@ -23,38 +23,32 @@ const ( |
|
|
func (store *FilerEtcStore) LoadConfiguration(ctx context.Context) (*iam_pb.S3ApiConfiguration, error) { |
|
|
func (store *FilerEtcStore) LoadConfiguration(ctx context.Context) (*iam_pb.S3ApiConfiguration, error) { |
|
|
s3cfg := &iam_pb.S3ApiConfiguration{} |
|
|
s3cfg := &iam_pb.S3ApiConfiguration{} |
|
|
|
|
|
|
|
|
// Try to load from multi-file structure first (identities directory)
|
|
|
|
|
|
hasMultiFile, err := store.loadFromMultiFile(ctx, s3cfg) |
|
|
|
|
|
|
|
|
// 1. Load from legacy single file (low priority)
|
|
|
|
|
|
content, foundLegacy, err := store.readInsideFiler(filer.IamConfigDirectory, IamLegacyIdentityFile) |
|
|
if err != nil { |
|
|
if err != nil { |
|
|
return s3cfg, err |
|
|
return s3cfg, err |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
if hasMultiFile { |
|
|
|
|
|
return s3cfg, nil |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
// Fallback to legacy single file
|
|
|
|
|
|
content, found, err := store.readInsideFiler(filer.IamConfigDirectory, IamLegacyIdentityFile) |
|
|
|
|
|
if err != nil { |
|
|
|
|
|
return s3cfg, err |
|
|
|
|
|
} |
|
|
|
|
|
if !found { |
|
|
|
|
|
// No config found at all
|
|
|
|
|
|
return s3cfg, nil |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
if len(content) > 0 { |
|
|
|
|
|
|
|
|
if foundLegacy && len(content) > 0 { |
|
|
if err := filer.ParseS3ConfigurationFromBytes(content, s3cfg); err != nil { |
|
|
if err := filer.ParseS3ConfigurationFromBytes(content, s3cfg); err != nil { |
|
|
glog.Errorf("Failed to parse legacy IAM configuration: %v", err) |
|
|
glog.Errorf("Failed to parse legacy IAM configuration: %v", err) |
|
|
return s3cfg, err |
|
|
return s3cfg, err |
|
|
} |
|
|
} |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
// Perform migration if we loaded legacy config
|
|
|
|
|
|
if err := store.migrateToMultiFile(ctx, s3cfg); err != nil { |
|
|
|
|
|
glog.Errorf("Failed to migrate IAM configuration to multi-file layout: %v", err) |
|
|
|
|
|
// Return the loaded config anyway, migration failed but we have data
|
|
|
|
|
|
return s3cfg, nil |
|
|
|
|
|
|
|
|
// 2. Load from multi-file structure (high priority, overrides legacy)
|
|
|
|
|
|
// This will merge identities into s3cfg
|
|
|
|
|
|
if _, err := store.loadFromMultiFile(ctx, s3cfg); err != nil { |
|
|
|
|
|
return s3cfg, err |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
// 3. Perform migration if we loaded legacy config
|
|
|
|
|
|
// This ensures that all identities (including legacy ones) are written to individual files
|
|
|
|
|
|
// and the legacy file is renamed.
|
|
|
|
|
|
if foundLegacy { |
|
|
|
|
|
if err := store.migrateToMultiFile(ctx, s3cfg); err != nil { |
|
|
|
|
|
glog.Errorf("Failed to migrate IAM configuration to multi-file layout: %v", err) |
|
|
|
|
|
return s3cfg, nil |
|
|
|
|
|
} |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
return s3cfg, nil |
|
|
return s3cfg, nil |
|
|
@ -63,6 +57,16 @@ func (store *FilerEtcStore) LoadConfiguration(ctx context.Context) (*iam_pb.S3Ap |
|
|
func (store *FilerEtcStore) loadFromMultiFile(ctx context.Context, s3cfg *iam_pb.S3ApiConfiguration) (bool, error) { |
|
|
func (store *FilerEtcStore) loadFromMultiFile(ctx context.Context, s3cfg *iam_pb.S3ApiConfiguration) (bool, error) { |
|
|
var hasIdentities bool |
|
|
var hasIdentities bool |
|
|
|
|
|
|
|
|
|
|
|
// Helper to find existing identity index
|
|
|
|
|
|
findIdentity := func(name string) int { |
|
|
|
|
|
for i, identity := range s3cfg.Identities { |
|
|
|
|
|
if identity.Name == name { |
|
|
|
|
|
return i |
|
|
|
|
|
} |
|
|
|
|
|
} |
|
|
|
|
|
return -1 |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
// 1. List identities
|
|
|
// 1. List identities
|
|
|
err := store.withFilerClient(func(client filer_pb.SeaweedFilerClient) error { |
|
|
err := store.withFilerClient(func(client filer_pb.SeaweedFilerClient) error { |
|
|
dir := filer.IamConfigDirectory + "/" + IamIdentitiesDirectory |
|
|
dir := filer.IamConfigDirectory + "/" + IamIdentitiesDirectory |
|
|
@ -78,17 +82,10 @@ func (store *FilerEtcStore) loadFromMultiFile(ctx context.Context, s3cfg *iam_pb |
|
|
} |
|
|
} |
|
|
hasIdentities = true |
|
|
hasIdentities = true |
|
|
|
|
|
|
|
|
// Read each identity file
|
|
|
|
|
|
// We can use the content from the entry if it's small (filer usually returns content for small files in ListEntries if configured,
|
|
|
|
|
|
// but ListEntries might not return content depending on implementation.
|
|
|
|
|
|
// filer.ListEntries calls ListEntriesRequest.
|
|
|
|
|
|
// Safest to read individually or check entry.Content
|
|
|
|
|
|
|
|
|
|
|
|
var content []byte |
|
|
var content []byte |
|
|
if len(entry.Content) > 0 { |
|
|
if len(entry.Content) > 0 { |
|
|
content = entry.Content |
|
|
content = entry.Content |
|
|
} else { |
|
|
} else { |
|
|
// Read file content
|
|
|
|
|
|
c, err := filer.ReadInsideFiler(client, dir, entry.Name) |
|
|
c, err := filer.ReadInsideFiler(client, dir, entry.Name) |
|
|
if err != nil { |
|
|
if err != nil { |
|
|
glog.Warningf("Failed to read identity file %s: %v", entry.Name, err) |
|
|
glog.Warningf("Failed to read identity file %s: %v", entry.Name, err) |
|
|
@ -103,7 +100,14 @@ func (store *FilerEtcStore) loadFromMultiFile(ctx context.Context, s3cfg *iam_pb |
|
|
glog.Warningf("Failed to unmarshal identity %s: %v", entry.Name, err) |
|
|
glog.Warningf("Failed to unmarshal identity %s: %v", entry.Name, err) |
|
|
continue |
|
|
continue |
|
|
} |
|
|
} |
|
|
s3cfg.Identities = append(s3cfg.Identities, identity) |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Merge logic: Overwrite existing or Append
|
|
|
|
|
|
idx := findIdentity(identity.Name) |
|
|
|
|
|
if idx != -1 { |
|
|
|
|
|
s3cfg.Identities[idx] = identity |
|
|
|
|
|
} else { |
|
|
|
|
|
s3cfg.Identities = append(s3cfg.Identities, identity) |
|
|
|
|
|
} |
|
|
} |
|
|
} |
|
|
} |
|
|
} |
|
|
return nil |
|
|
return nil |
|
|
@ -119,15 +123,12 @@ func (store *FilerEtcStore) loadFromMultiFile(ctx context.Context, s3cfg *iam_pb |
|
|
return false, err |
|
|
return false, err |
|
|
} |
|
|
} |
|
|
if found && len(content) > 0 { |
|
|
if found && len(content) > 0 { |
|
|
// We use a temporary struct to load just the non-identity parts,
|
|
|
|
|
|
// but ParseS3ConfigurationFromBytes expects full config.
|
|
|
|
|
|
// It's robust to missing fields.
|
|
|
|
|
|
tempCfg := &iam_pb.S3ApiConfiguration{} |
|
|
tempCfg := &iam_pb.S3ApiConfiguration{} |
|
|
if err := filer.ParseS3ConfigurationFromBytes(content, tempCfg); err == nil { |
|
|
if err := filer.ParseS3ConfigurationFromBytes(content, tempCfg); err == nil { |
|
|
|
|
|
// Overwrite accounts from configuration.json (high priority)
|
|
|
s3cfg.Accounts = tempCfg.Accounts |
|
|
s3cfg.Accounts = tempCfg.Accounts |
|
|
// Copy other fields if any
|
|
|
|
|
|
} |
|
|
} |
|
|
return true, nil // Found configuration file, so it is multi-file
|
|
|
|
|
|
|
|
|
return true, nil |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
return hasIdentities, nil |
|
|
return hasIdentities, nil |
|
|
|