mirror of
https://github.com/minio/minio.git
synced 2025-01-11 23:13:23 -05:00
Move all IAM storage functionality into iam store type (#13567)
This reverts commit 091a7ae359
.
- Ensure all actions accessing storage lock properly.
- Behavior change: policies can be deleted only when they
are not associated with any active credentials.
Also adds fix for accidental canned policy removal that was present in the
reverted version of the change.
This commit is contained in:
parent
ca2b288a4b
commit
ecd54b4cba
@ -103,6 +103,12 @@ func toAdminAPIErr(ctx context.Context, err error) APIError {
|
|||||||
Description: err.Error(),
|
Description: err.Error(),
|
||||||
HTTPStatusCode: http.StatusServiceUnavailable,
|
HTTPStatusCode: http.StatusServiceUnavailable,
|
||||||
}
|
}
|
||||||
|
case errors.Is(err, errPolicyInUse):
|
||||||
|
apiErr = APIError{
|
||||||
|
Code: "XMinioAdminPolicyInUse",
|
||||||
|
Description: "The policy cannot be removed, as it is in use",
|
||||||
|
HTTPStatusCode: http.StatusBadRequest,
|
||||||
|
}
|
||||||
case errors.Is(err, kes.ErrKeyExists):
|
case errors.Is(err, kes.ErrKeyExists):
|
||||||
apiErr = APIError{
|
apiErr = APIError{
|
||||||
Code: "XMinioKMSKeyExists",
|
Code: "XMinioKMSKeyExists",
|
||||||
|
@ -258,10 +258,20 @@ func (s *TestSuiteIAM) TestPolicyCreate(c *check) {
|
|||||||
c.Fatalf("policy was missing!")
|
c.Fatalf("policy was missing!")
|
||||||
}
|
}
|
||||||
|
|
||||||
// 5. Check that policy can be deleted.
|
// 5. Check that policy cannot be deleted when attached to a user.
|
||||||
|
err = s.adm.RemoveCannedPolicy(ctx, policy)
|
||||||
|
if err == nil {
|
||||||
|
c.Fatalf("policy could be unexpectedly deleted!")
|
||||||
|
}
|
||||||
|
|
||||||
|
// 6. Delete the user and then delete the policy.
|
||||||
|
err = s.adm.RemoveUser(ctx, accessKey)
|
||||||
|
if err != nil {
|
||||||
|
c.Fatalf("user could not be deleted: %v", err)
|
||||||
|
}
|
||||||
err = s.adm.RemoveCannedPolicy(ctx, policy)
|
err = s.adm.RemoveCannedPolicy(ctx, policy)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
c.Fatalf("policy delete err: %v", err)
|
c.Fatalf("policy del err: %v", err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -627,7 +637,8 @@ func (c *check) mustListObjects(ctx context.Context, client *minio.Client, bucke
|
|||||||
res := client.ListObjects(ctx, bucket, minio.ListObjectsOptions{})
|
res := client.ListObjects(ctx, bucket, minio.ListObjectsOptions{})
|
||||||
v, ok := <-res
|
v, ok := <-res
|
||||||
if ok && v.Err != nil {
|
if ok && v.Err != nil {
|
||||||
c.Fatalf("user was unable to list unexpectedly!")
|
msg := fmt.Sprintf("user was unable to list: %v", v.Err)
|
||||||
|
c.Fatalf(msg)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -27,22 +27,37 @@ import (
|
|||||||
|
|
||||||
type iamDummyStore struct {
|
type iamDummyStore struct {
|
||||||
sync.RWMutex
|
sync.RWMutex
|
||||||
|
*iamCache
|
||||||
|
usersSysType UsersSysType
|
||||||
}
|
}
|
||||||
|
|
||||||
func (ids *iamDummyStore) lock() {
|
func newIAMDummyStore(usersSysType UsersSysType) *iamDummyStore {
|
||||||
|
return &iamDummyStore{
|
||||||
|
iamCache: newIamCache(),
|
||||||
|
usersSysType: usersSysType,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ids *iamDummyStore) rlock() *iamCache {
|
||||||
|
ids.RLock()
|
||||||
|
return ids.iamCache
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ids *iamDummyStore) runlock() {
|
||||||
|
ids.RUnlock()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ids *iamDummyStore) lock() *iamCache {
|
||||||
ids.Lock()
|
ids.Lock()
|
||||||
|
return ids.iamCache
|
||||||
}
|
}
|
||||||
|
|
||||||
func (ids *iamDummyStore) unlock() {
|
func (ids *iamDummyStore) unlock() {
|
||||||
ids.Unlock()
|
ids.Unlock()
|
||||||
}
|
}
|
||||||
|
|
||||||
func (ids *iamDummyStore) rlock() {
|
func (ids *iamDummyStore) getUsersSysType() UsersSysType {
|
||||||
ids.RLock()
|
return ids.usersSysType
|
||||||
}
|
|
||||||
|
|
||||||
func (ids *iamDummyStore) runlock() {
|
|
||||||
ids.RUnlock()
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (ids *iamDummyStore) migrateBackendFormat(context.Context) error {
|
func (ids *iamDummyStore) migrateBackendFormat(context.Context) error {
|
||||||
|
@ -62,27 +62,37 @@ func extractPathPrefixAndSuffix(s string, prefix string, suffix string) string {
|
|||||||
type IAMEtcdStore struct {
|
type IAMEtcdStore struct {
|
||||||
sync.RWMutex
|
sync.RWMutex
|
||||||
|
|
||||||
|
*iamCache
|
||||||
|
|
||||||
|
usersSysType UsersSysType
|
||||||
|
|
||||||
client *etcd.Client
|
client *etcd.Client
|
||||||
}
|
}
|
||||||
|
|
||||||
func newIAMEtcdStore(client *etcd.Client) *IAMEtcdStore {
|
func newIAMEtcdStore(client *etcd.Client, usersSysType UsersSysType) *IAMEtcdStore {
|
||||||
return &IAMEtcdStore{client: client}
|
return &IAMEtcdStore{client: client, usersSysType: usersSysType}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (ies *IAMEtcdStore) lock() {
|
func (ies *IAMEtcdStore) rlock() *iamCache {
|
||||||
|
ies.RLock()
|
||||||
|
return ies.iamCache
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ies *IAMEtcdStore) runlock() {
|
||||||
|
ies.RUnlock()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ies *IAMEtcdStore) lock() *iamCache {
|
||||||
ies.Lock()
|
ies.Lock()
|
||||||
|
return ies.iamCache
|
||||||
}
|
}
|
||||||
|
|
||||||
func (ies *IAMEtcdStore) unlock() {
|
func (ies *IAMEtcdStore) unlock() {
|
||||||
ies.Unlock()
|
ies.Unlock()
|
||||||
}
|
}
|
||||||
|
|
||||||
func (ies *IAMEtcdStore) rlock() {
|
func (ies *IAMEtcdStore) getUsersSysType() UsersSysType {
|
||||||
ies.RLock()
|
return ies.usersSysType
|
||||||
}
|
|
||||||
|
|
||||||
func (ies *IAMEtcdStore) runlock() {
|
|
||||||
ies.RUnlock()
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (ies *IAMEtcdStore) saveIAMConfig(ctx context.Context, item interface{}, itemPath string, opts ...options) error {
|
func (ies *IAMEtcdStore) saveIAMConfig(ctx context.Context, item interface{}, itemPath string, opts ...options) error {
|
||||||
@ -244,6 +254,8 @@ func (ies *IAMEtcdStore) migrateToV1(ctx context.Context) error {
|
|||||||
|
|
||||||
// Should be called under config migration lock
|
// Should be called under config migration lock
|
||||||
func (ies *IAMEtcdStore) migrateBackendFormat(ctx context.Context) error {
|
func (ies *IAMEtcdStore) migrateBackendFormat(ctx context.Context) error {
|
||||||
|
ies.Lock()
|
||||||
|
defer ies.Unlock()
|
||||||
return ies.migrateToV1(ctx)
|
return ies.migrateToV1(ctx)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -260,7 +272,7 @@ func (ies *IAMEtcdStore) loadPolicyDoc(ctx context.Context, policy string, m map
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (ies *IAMEtcdStore) getPolicyDoc(ctx context.Context, kvs *mvccpb.KeyValue, m map[string]iampolicy.Policy) error {
|
func (ies *IAMEtcdStore) getPolicyDocKV(ctx context.Context, kvs *mvccpb.KeyValue, m map[string]iampolicy.Policy) error {
|
||||||
var p iampolicy.Policy
|
var p iampolicy.Policy
|
||||||
err := getIAMConfig(&p, kvs.Value, string(kvs.Key))
|
err := getIAMConfig(&p, kvs.Value, string(kvs.Key))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@ -286,14 +298,14 @@ func (ies *IAMEtcdStore) loadPolicyDocs(ctx context.Context, m map[string]iampol
|
|||||||
|
|
||||||
// Parse all values to construct the policies data model.
|
// Parse all values to construct the policies data model.
|
||||||
for _, kvs := range r.Kvs {
|
for _, kvs := range r.Kvs {
|
||||||
if err = ies.getPolicyDoc(ctx, kvs, m); err != nil && err != errNoSuchPolicy {
|
if err = ies.getPolicyDocKV(ctx, kvs, m); err != nil && err != errNoSuchPolicy {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (ies *IAMEtcdStore) getUser(ctx context.Context, userkv *mvccpb.KeyValue, userType IAMUserType, m map[string]auth.Credentials, basePrefix string) error {
|
func (ies *IAMEtcdStore) getUserKV(ctx context.Context, userkv *mvccpb.KeyValue, userType IAMUserType, m map[string]auth.Credentials, basePrefix string) error {
|
||||||
var u UserIdentity
|
var u UserIdentity
|
||||||
err := getIAMConfig(&u, userkv.Value, string(userkv.Key))
|
err := getIAMConfig(&u, userkv.Value, string(userkv.Key))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@ -355,7 +367,7 @@ func (ies *IAMEtcdStore) loadUsers(ctx context.Context, userType IAMUserType, m
|
|||||||
|
|
||||||
// Parse all users values to create the proper data model
|
// Parse all users values to create the proper data model
|
||||||
for _, userKv := range r.Kvs {
|
for _, userKv := range r.Kvs {
|
||||||
if err = ies.getUser(ctx, userKv, userType, m, basePrefix); err != nil && err != errNoSuchUser {
|
if err = ies.getUserKV(ctx, userKv, userType, m, basePrefix); err != nil && err != errNoSuchUser {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -34,30 +34,44 @@ import (
|
|||||||
|
|
||||||
// IAMObjectStore implements IAMStorageAPI
|
// IAMObjectStore implements IAMStorageAPI
|
||||||
type IAMObjectStore struct {
|
type IAMObjectStore struct {
|
||||||
// Protect assignment to objAPI
|
// Protect access to storage within the current server.
|
||||||
sync.RWMutex
|
sync.RWMutex
|
||||||
|
|
||||||
|
*iamCache
|
||||||
|
|
||||||
|
usersSysType UsersSysType
|
||||||
|
|
||||||
objAPI ObjectLayer
|
objAPI ObjectLayer
|
||||||
}
|
}
|
||||||
|
|
||||||
func newIAMObjectStore(objAPI ObjectLayer) *IAMObjectStore {
|
func newIAMObjectStore(objAPI ObjectLayer, usersSysType UsersSysType) *IAMObjectStore {
|
||||||
return &IAMObjectStore{objAPI: objAPI}
|
return &IAMObjectStore{
|
||||||
|
iamCache: newIamCache(),
|
||||||
|
objAPI: objAPI,
|
||||||
|
usersSysType: usersSysType,
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (iamOS *IAMObjectStore) lock() {
|
func (iamOS *IAMObjectStore) rlock() *iamCache {
|
||||||
|
iamOS.RLock()
|
||||||
|
return iamOS.iamCache
|
||||||
|
}
|
||||||
|
|
||||||
|
func (iamOS *IAMObjectStore) runlock() {
|
||||||
|
iamOS.RUnlock()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (iamOS *IAMObjectStore) lock() *iamCache {
|
||||||
iamOS.Lock()
|
iamOS.Lock()
|
||||||
|
return iamOS.iamCache
|
||||||
}
|
}
|
||||||
|
|
||||||
func (iamOS *IAMObjectStore) unlock() {
|
func (iamOS *IAMObjectStore) unlock() {
|
||||||
iamOS.Unlock()
|
iamOS.Unlock()
|
||||||
}
|
}
|
||||||
|
|
||||||
func (iamOS *IAMObjectStore) rlock() {
|
func (iamOS *IAMObjectStore) getUsersSysType() UsersSysType {
|
||||||
iamOS.RLock()
|
return iamOS.usersSysType
|
||||||
}
|
|
||||||
|
|
||||||
func (iamOS *IAMObjectStore) runlock() {
|
|
||||||
iamOS.RUnlock()
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Migrate users directory in a single scan.
|
// Migrate users directory in a single scan.
|
||||||
@ -182,6 +196,8 @@ func (iamOS *IAMObjectStore) migrateToV1(ctx context.Context) error {
|
|||||||
|
|
||||||
// Should be called under config migration lock
|
// Should be called under config migration lock
|
||||||
func (iamOS *IAMObjectStore) migrateBackendFormat(ctx context.Context) error {
|
func (iamOS *IAMObjectStore) migrateBackendFormat(ctx context.Context) error {
|
||||||
|
iamOS.Lock()
|
||||||
|
defer iamOS.Unlock()
|
||||||
return iamOS.migrateToV1(ctx)
|
return iamOS.migrateToV1(ctx)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
1716
cmd/iam-store.go
Normal file
1716
cmd/iam-store.go
Normal file
File diff suppressed because it is too large
Load Diff
1740
cmd/iam.go
1740
cmd/iam.go
File diff suppressed because it is too large
Load Diff
@ -20,7 +20,6 @@ package cmd
|
|||||||
import (
|
import (
|
||||||
"bytes"
|
"bytes"
|
||||||
"context"
|
"context"
|
||||||
"crypto/rand"
|
|
||||||
"crypto/tls"
|
"crypto/tls"
|
||||||
"encoding/base64"
|
"encoding/base64"
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
@ -355,21 +354,13 @@ func (c *SiteReplicationSys) AddPeerClusters(ctx context.Context, sites []madmin
|
|||||||
|
|
||||||
// Generate a secret key for the service account.
|
// Generate a secret key for the service account.
|
||||||
var secretKey string
|
var secretKey string
|
||||||
{
|
_, secretKey, err := auth.GenerateCredentials()
|
||||||
secretKeyBuf := make([]byte, 40)
|
|
||||||
n, err := rand.Read(secretKeyBuf)
|
|
||||||
if err == nil && n != 40 {
|
|
||||||
err = fmt.Errorf("Unable to read 40 random bytes to generate secret key")
|
|
||||||
}
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return madmin.ReplicateAddStatus{}, SRError{
|
return madmin.ReplicateAddStatus{}, SRError{
|
||||||
Cause: err,
|
Cause: err,
|
||||||
Code: ErrInternalError,
|
Code: ErrInternalError,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
secretKey = strings.Replace(string([]byte(base64.StdEncoding.EncodeToString(secretKeyBuf))[:40]),
|
|
||||||
"/", "+", -1)
|
|
||||||
}
|
|
||||||
|
|
||||||
svcCred, err := globalIAMSys.NewServiceAccount(ctx, sites[selfIdx].AccessKey, nil, newServiceAccountOpts{
|
svcCred, err := globalIAMSys.NewServiceAccount(ctx, sites[selfIdx].AccessKey, nil, newServiceAccountOpts{
|
||||||
accessKey: siteReplicatorSvcAcc,
|
accessKey: siteReplicatorSvcAcc,
|
||||||
@ -1270,9 +1261,7 @@ func (c *SiteReplicationSys) getAdminClient(ctx context.Context, deploymentID st
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (c *SiteReplicationSys) getPeerCreds() (*auth.Credentials, error) {
|
func (c *SiteReplicationSys) getPeerCreds() (*auth.Credentials, error) {
|
||||||
globalIAMSys.store.rlock()
|
creds, ok := globalIAMSys.store.GetUser(c.state.ServiceAccountAccessKey)
|
||||||
defer globalIAMSys.store.runlock()
|
|
||||||
creds, ok := globalIAMSys.iamUsersMap[c.state.ServiceAccountAccessKey]
|
|
||||||
if !ok {
|
if !ok {
|
||||||
return nil, errors.New("site replication service account not found!")
|
return nil, errors.New("site replication service account not found!")
|
||||||
}
|
}
|
||||||
|
@ -95,6 +95,10 @@ func (s *TestSuiteIAM) TestSTS(c *check) {
|
|||||||
c.Fatalf("Unable to set policy: %v", err)
|
c.Fatalf("Unable to set policy: %v", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// confirm that the user is able to access the bucket
|
||||||
|
uClient := s.getUserClient(c, accessKey, secretKey, "")
|
||||||
|
c.mustListObjects(ctx, uClient, bucket)
|
||||||
|
|
||||||
assumeRole := cr.STSAssumeRole{
|
assumeRole := cr.STSAssumeRole{
|
||||||
Client: s.TestSuiteCommon.client,
|
Client: s.TestSuiteCommon.client,
|
||||||
STSEndpoint: s.endPoint,
|
STSEndpoint: s.endPoint,
|
||||||
|
@ -81,6 +81,9 @@ var errGroupNotEmpty = errors.New("Specified group is not empty - cannot remove
|
|||||||
// error returned in IAM subsystem when policy doesn't exist.
|
// error returned in IAM subsystem when policy doesn't exist.
|
||||||
var errNoSuchPolicy = errors.New("Specified canned policy does not exist")
|
var errNoSuchPolicy = errors.New("Specified canned policy does not exist")
|
||||||
|
|
||||||
|
// error returned when policy to be deleted is in use.
|
||||||
|
var errPolicyInUse = errors.New("Specified policy is in use and cannot be deleted.")
|
||||||
|
|
||||||
// error returned in IAM subsystem when an external users systems is configured.
|
// error returned in IAM subsystem when an external users systems is configured.
|
||||||
var errIAMActionNotAllowed = errors.New("Specified IAM action is not allowed")
|
var errIAMActionNotAllowed = errors.New("Specified IAM action is not allowed")
|
||||||
|
|
||||||
|
2
go.mod
2
go.mod
@ -50,7 +50,7 @@ require (
|
|||||||
github.com/minio/madmin-go v1.1.11-0.20211102182201-e51fd3d6b104
|
github.com/minio/madmin-go v1.1.11-0.20211102182201-e51fd3d6b104
|
||||||
github.com/minio/minio-go/v7 v7.0.15
|
github.com/minio/minio-go/v7 v7.0.15
|
||||||
github.com/minio/parquet-go v1.0.0
|
github.com/minio/parquet-go v1.0.0
|
||||||
github.com/minio/pkg v1.1.5
|
github.com/minio/pkg v1.1.6-0.20211103212545-951bbd71498c
|
||||||
github.com/minio/selfupdate v0.3.1
|
github.com/minio/selfupdate v0.3.1
|
||||||
github.com/minio/sha256-simd v1.0.0
|
github.com/minio/sha256-simd v1.0.0
|
||||||
github.com/minio/simdjson-go v0.2.1
|
github.com/minio/simdjson-go v0.2.1
|
||||||
|
4
go.sum
4
go.sum
@ -1077,6 +1077,10 @@ github.com/minio/pkg v1.0.11/go.mod h1:32x/3OmGB0EOi1N+3ggnp+B5VFkSBBB9svPMVfpnf
|
|||||||
github.com/minio/pkg v1.1.3/go.mod h1:32x/3OmGB0EOi1N+3ggnp+B5VFkSBBB9svPMVfpnf14=
|
github.com/minio/pkg v1.1.3/go.mod h1:32x/3OmGB0EOi1N+3ggnp+B5VFkSBBB9svPMVfpnf14=
|
||||||
github.com/minio/pkg v1.1.5 h1:phwKkJBQdVLyxOXC3RChPVGLtebplzQJ5jJ3l/HBvnk=
|
github.com/minio/pkg v1.1.5 h1:phwKkJBQdVLyxOXC3RChPVGLtebplzQJ5jJ3l/HBvnk=
|
||||||
github.com/minio/pkg v1.1.5/go.mod h1:32x/3OmGB0EOi1N+3ggnp+B5VFkSBBB9svPMVfpnf14=
|
github.com/minio/pkg v1.1.5/go.mod h1:32x/3OmGB0EOi1N+3ggnp+B5VFkSBBB9svPMVfpnf14=
|
||||||
|
github.com/minio/pkg v1.1.6-0.20211102234044-cd6b7b169e31 h1:nZkTtdcp4JgClBFI+mZJNO1J+8bEpcrOumdsbgdtF0A=
|
||||||
|
github.com/minio/pkg v1.1.6-0.20211102234044-cd6b7b169e31/go.mod h1:32x/3OmGB0EOi1N+3ggnp+B5VFkSBBB9svPMVfpnf14=
|
||||||
|
github.com/minio/pkg v1.1.6-0.20211103212545-951bbd71498c h1:zP0nEhOBjJRu6fP8nrNMUoGVZGIHbFKY1Ln5V/6Djbg=
|
||||||
|
github.com/minio/pkg v1.1.6-0.20211103212545-951bbd71498c/go.mod h1:32x/3OmGB0EOi1N+3ggnp+B5VFkSBBB9svPMVfpnf14=
|
||||||
github.com/minio/selfupdate v0.3.1 h1:BWEFSNnrZVMUWXbXIgLDNDjbejkmpAmZvy/nCz1HlEs=
|
github.com/minio/selfupdate v0.3.1 h1:BWEFSNnrZVMUWXbXIgLDNDjbejkmpAmZvy/nCz1HlEs=
|
||||||
github.com/minio/selfupdate v0.3.1/go.mod h1:b8ThJzzH7u2MkF6PcIra7KaXO9Khf6alWPvMSyTDCFM=
|
github.com/minio/selfupdate v0.3.1/go.mod h1:b8ThJzzH7u2MkF6PcIra7KaXO9Khf6alWPvMSyTDCFM=
|
||||||
github.com/minio/sha256-simd v0.1.1/go.mod h1:B5e1o+1/KgNmWrSQK08Y6Z1Vb5pwIktudl0J58iy0KM=
|
github.com/minio/sha256-simd v0.1.1/go.mod h1:B5e1o+1/KgNmWrSQK08Y6Z1Vb5pwIktudl0J58iy0KM=
|
||||||
|
Loading…
Reference in New Issue
Block a user