mirror of
https://github.com/minio/minio.git
synced 2025-01-11 15:03:22 -05:00
feat: Allow at most one claim based OpenID IDP (#16145)
This commit is contained in:
parent
be92cf5959
commit
87cbd41265
@ -1187,6 +1187,7 @@ func (a adminAPIHandlers) AccountInfoHandler(w http.ResponseWriter, r *http.Requ
|
|||||||
}
|
}
|
||||||
|
|
||||||
roleArn := iampolicy.Args{Claims: claims}.GetRoleArn()
|
roleArn := iampolicy.Args{Claims: claims}.GetRoleArn()
|
||||||
|
policySetFromClaims, hasPolicyClaim := iampolicy.GetPoliciesFromClaims(claims, iamPolicyClaimNameOpenID())
|
||||||
var effectivePolicy iampolicy.Policy
|
var effectivePolicy iampolicy.Policy
|
||||||
|
|
||||||
var buf []byte
|
var buf []byte
|
||||||
@ -1198,16 +1199,19 @@ func (a adminAPIHandlers) AccountInfoHandler(w http.ResponseWriter, r *http.Requ
|
|||||||
break
|
break
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
case roleArn != "":
|
case roleArn != "":
|
||||||
_, policy, err := globalIAMSys.GetRolePolicy(roleArn)
|
_, policy, err := globalIAMSys.GetRolePolicy(roleArn)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.LogIf(ctx, err)
|
|
||||||
writeErrorResponseJSON(ctx, w, toAdminAPIErr(ctx, err), r.URL)
|
writeErrorResponseJSON(ctx, w, toAdminAPIErr(ctx, err), r.URL)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
policySlice := newMappedPolicy(policy).toSlice()
|
policySlice := newMappedPolicy(policy).toSlice()
|
||||||
effectivePolicy = globalIAMSys.GetCombinedPolicy(policySlice...)
|
effectivePolicy = globalIAMSys.GetCombinedPolicy(policySlice...)
|
||||||
|
|
||||||
|
case hasPolicyClaim:
|
||||||
|
effectivePolicy = globalIAMSys.GetCombinedPolicy(policySetFromClaims.ToSlice()...)
|
||||||
|
|
||||||
default:
|
default:
|
||||||
policies, err := globalIAMSys.PolicyDBGet(accountName, false, cred.Groups...)
|
policies, err := globalIAMSys.PolicyDBGet(accountName, false, cred.Groups...)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -343,17 +343,23 @@ func (sts *stsAPIHandlers) AssumeRoleWithSSO(w http.ResponseWriter, r *http.Requ
|
|||||||
|
|
||||||
accessToken := r.Form.Get(stsWebIdentityAccessToken)
|
accessToken := r.Form.Get(stsWebIdentityAccessToken)
|
||||||
|
|
||||||
|
// RoleARN parameter processing: If a role ARN is given in the request, we
|
||||||
|
// use that and validate the authentication request. If not, we assume this
|
||||||
|
// is an STS request for a claim based IDP (if one is present) and set
|
||||||
|
// roleArn = openid.DummyRoleARN.
|
||||||
|
//
|
||||||
|
// Currently, we do not support multiple claim based IDPs, as there is no
|
||||||
|
// defined parameter to disambiguate the intended IDP in this STS request.
|
||||||
roleArn := openid.DummyRoleARN
|
roleArn := openid.DummyRoleARN
|
||||||
if globalIAMSys.HasRolePolicy() {
|
roleArnStr := r.Form.Get(stsRoleArn)
|
||||||
|
if roleArnStr != "" {
|
||||||
var err error
|
var err error
|
||||||
roleArnStr := r.Form.Get(stsRoleArn)
|
|
||||||
roleArn, _, err = globalIAMSys.GetRolePolicy(roleArnStr)
|
roleArn, _, err = globalIAMSys.GetRolePolicy(roleArnStr)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
writeSTSErrorResponse(ctx, w, true, ErrSTSInvalidParameterValue,
|
writeSTSErrorResponse(ctx, w, true, ErrSTSInvalidParameterValue,
|
||||||
fmt.Errorf("Error processing %s parameter: %v", stsRoleArn, err))
|
fmt.Errorf("Error processing %s parameter: %v", stsRoleArn, err))
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Validate JWT; check clientID in claims matches the one associated with the roleArn
|
// Validate JWT; check clientID in claims matches the one associated with the roleArn
|
||||||
@ -376,7 +382,7 @@ func (sts *stsAPIHandlers) AssumeRoleWithSSO(w http.ResponseWriter, r *http.Requ
|
|||||||
}
|
}
|
||||||
|
|
||||||
var policyName string
|
var policyName string
|
||||||
if globalIAMSys.HasRolePolicy() {
|
if roleArnStr != "" && globalIAMSys.HasRolePolicy() {
|
||||||
// If roleArn is used, we set it as a claim, and use the
|
// If roleArn is used, we set it as a claim, and use the
|
||||||
// associated policy when credentials are used.
|
// associated policy when credentials are used.
|
||||||
claims[roleArnClaim] = roleArn.String()
|
claims[roleArnClaim] = roleArn.String()
|
||||||
|
@ -1563,7 +1563,7 @@ func (s *TestSuiteIAM) TestOpenIDSTSWithRolePolicyWithPolVar(c *check, roleARN s
|
|||||||
c.mustNotListObjects(ctx, lisaClient, "other")
|
c.mustNotListObjects(ctx, lisaClient, "other")
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestIAMWithOpenIDMultipleConfigsValidation(t *testing.T) {
|
func TestIAMWithOpenIDMultipleConfigsValidation1(t *testing.T) {
|
||||||
openIDServer := os.Getenv(EnvTestOpenIDServer)
|
openIDServer := os.Getenv(EnvTestOpenIDServer)
|
||||||
openIDServer2 := os.Getenv(EnvTestOpenIDServer2)
|
openIDServer2 := os.Getenv(EnvTestOpenIDServer2)
|
||||||
if openIDServer == "" || openIDServer2 == "" {
|
if openIDServer == "" || openIDServer2 == "" {
|
||||||
@ -1576,6 +1576,38 @@ func TestIAMWithOpenIDMultipleConfigsValidation(t *testing.T) {
|
|||||||
"readwrite",
|
"readwrite",
|
||||||
}
|
}
|
||||||
|
|
||||||
|
for i, testCase := range iamTestSuites {
|
||||||
|
t.Run(
|
||||||
|
fmt.Sprintf("Test: %d, ServerType: %s", i+1, testCase.ServerTypeDescription),
|
||||||
|
func(t *testing.T) {
|
||||||
|
c := &check{t, testCase.serverType}
|
||||||
|
suite := testCase
|
||||||
|
|
||||||
|
suite.SetUpSuite(c)
|
||||||
|
defer suite.TearDownSuite(c)
|
||||||
|
|
||||||
|
err := suite.SetUpOpenIDs(c, testApps, rolePolicies)
|
||||||
|
if err != nil {
|
||||||
|
c.Fatalf("config with 1 claim based and 1 role based provider should pass but got: %v", err)
|
||||||
|
}
|
||||||
|
},
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestIAMWithOpenIDMultipleConfigsValidation2(t *testing.T) {
|
||||||
|
openIDServer := os.Getenv(EnvTestOpenIDServer)
|
||||||
|
openIDServer2 := os.Getenv(EnvTestOpenIDServer2)
|
||||||
|
if openIDServer == "" || openIDServer2 == "" {
|
||||||
|
t.Skip("Skipping OpenID test as enough OpenID servers are not provided.")
|
||||||
|
}
|
||||||
|
testApps := testClientApps
|
||||||
|
|
||||||
|
rolePolicies := []string{
|
||||||
|
"", // Treated as claim-based provider as no role policy is given.
|
||||||
|
"", // Treated as claim-based provider as no role policy is given.
|
||||||
|
}
|
||||||
|
|
||||||
for i, testCase := range iamTestSuites {
|
for i, testCase := range iamTestSuites {
|
||||||
t.Run(
|
t.Run(
|
||||||
fmt.Sprintf("Test: %d, ServerType: %s", i+1, testCase.ServerTypeDescription),
|
fmt.Sprintf("Test: %d, ServerType: %s", i+1, testCase.ServerTypeDescription),
|
||||||
@ -1588,7 +1620,7 @@ func TestIAMWithOpenIDMultipleConfigsValidation(t *testing.T) {
|
|||||||
|
|
||||||
err := suite.SetUpOpenIDs(c, testApps, rolePolicies)
|
err := suite.SetUpOpenIDs(c, testApps, rolePolicies)
|
||||||
if err == nil {
|
if err == nil {
|
||||||
c.Fatal("config with both claim based and role policy based providers should fail")
|
c.Fatalf("config with 2 claim based provider should fail")
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
)
|
)
|
||||||
|
@ -37,7 +37,7 @@ MINIO_IDENTITY_OPENID_REDIRECT_URI (string) [DEPRECATED use env 'MIN
|
|||||||
|
|
||||||
Either `MINIO_IDENTITY_OPENID_ROLE_POLICY` (recommended) or `MINIO_IDENTITY_OPENID_CLAIM_NAME` must be specified but not both. See the section Access Control Policies to understand the differences between the two.
|
Either `MINIO_IDENTITY_OPENID_ROLE_POLICY` (recommended) or `MINIO_IDENTITY_OPENID_CLAIM_NAME` must be specified but not both. See the section Access Control Policies to understand the differences between the two.
|
||||||
|
|
||||||
With role policies, it is possible to specify multiple OpenID provider configurations - this is useful to integrate multiple OpenID client applications to interact with object storage.
|
**NOTE**: When configuring multiple OpenID based authentication providers on a MinIO cluster, any number of Role Policy based providers may be configured, and at most one JWT Claim based provider may be configured.
|
||||||
|
|
||||||
<details><summary>Example 1: Two role policy providers</summary>
|
<details><summary>Example 1: Two role policy providers</summary>
|
||||||
|
|
||||||
@ -80,13 +80,13 @@ MINIO_IDENTITY_OPENID_CLAIM_NAME="groups"
|
|||||||
|
|
||||||
</details>
|
</details>
|
||||||
|
|
||||||
## Access Control Policies
|
## Specifying Access Control with IAM Policies
|
||||||
|
|
||||||
MinIO's AssumeRoleWithWebIdentity supports specifying access control policies in two ways:
|
The STS API authenticates the user by verifying the JWT provided in the request. However access to object storage resources are controlled via named IAM policies defined in the MinIO instance. Once authenticated via the STS API, the MinIO server applies one or more IAM policies to the generated credentials. MinIO's AssumeRoleWithWebIdentity implementation supports specifying IAM policies in two ways:
|
||||||
|
|
||||||
1. Role Policy (Recommended): When specified, all users authenticating via this API are authorized to (only) use the specified role policy. The policy to associate with such users is specified when configuring OpenID provider in the server, via the `role_policy` configuration parameter or the `MINIO_IDENTITY_OPENID_ROLE_POLICY` environment variable. The value is a comma-separated list of IAM access policy names already defined in the server. In this situation, the server prints a role ARN at startup that must be specified as a `RoleARN` API request parameter in the STS AssumeRoleWithWebIdentity API call. When using Role Policies, multiple OpenID providers and/or client applications (with unique client IDs) may be configured with independent role policies. Each configuration is assigned a unique RoleARN by the MinIO server and this is used to select the policies to apply to temporary credentials generated in the AssumeRoleWithWebIdentity call.
|
1. Role Policy (Recommended): When specified as part of the OpenID provider configuration, all users authenticating via this provider are authorized to (only) use the specified role policy. The policy to associate with such users is specified via the `role_policy` configuration parameter or the `MINIO_IDENTITY_OPENID_ROLE_POLICY` environment variable. The value is a comma-separated list of IAM access policy names already defined in the server. In this situation, the server prints a role ARN at startup that must be specified as a `RoleARN` API request parameter in the STS AssumeRoleWithWebIdentity API call. When using Role Policies, multiple OpenID providers and/or client applications (with unique client IDs) may be configured with independent role policies. Each configuration is assigned a unique RoleARN by the MinIO server and this is used to select the policies to apply to temporary credentials generated in the AssumeRoleWithWebIdentity call.
|
||||||
|
|
||||||
2. `id_token` claims: When the role policy is not configured, MinIO looks for a specific claim in the `id_token` (JWT) returned by the OpenID provider. The default claim is `policy` and can be overridden by the `claim_name` configuration parameter or the `MINIO_IDENTITY_OPENID_CLAIM_NAME` environment variable. The claim value can be a string (comma-separated list) or an array of IAM access policy names defined in the server. A `RoleARN` API request parameter *must not* be specified in the STS AssumeRoleWithWebIdentity API call.
|
2. `id_token` claims: When the role policy is not configured, MinIO looks for a specific claim in the `id_token` (JWT) returned by the OpenID provider in the STS request. The default claim is `policy` and can be overridden by the `claim_name` configuration parameter or the `MINIO_IDENTITY_OPENID_CLAIM_NAME` environment variable. The claim value can be a string (comma-separated list) or an array of IAM access policy names defined in the server. A `RoleARN` API request parameter *must not* be specified in the STS AssumeRoleWithWebIdentity API call.
|
||||||
|
|
||||||
## API Request Parameters
|
## API Request Parameters
|
||||||
|
|
||||||
|
@ -204,10 +204,7 @@ func LookupConfig(s config.Config, transport http.RoundTripper, closeRespFn func
|
|||||||
closeRespFn: closeRespFn,
|
closeRespFn: closeRespFn,
|
||||||
}
|
}
|
||||||
|
|
||||||
var (
|
seenClientIDs := set.NewStringSet()
|
||||||
hasLegacyPolicyMapping = false
|
|
||||||
seenClientIDs = set.NewStringSet()
|
|
||||||
)
|
|
||||||
|
|
||||||
deprecatedKeys := []string{JwksURL}
|
deprecatedKeys := []string{JwksURL}
|
||||||
|
|
||||||
@ -376,9 +373,8 @@ func LookupConfig(s config.Config, transport http.RoundTripper, closeRespFn func
|
|||||||
arnKey := p.roleArn
|
arnKey := p.roleArn
|
||||||
if p.RolePolicy == "" {
|
if p.RolePolicy == "" {
|
||||||
arnKey = DummyRoleARN
|
arnKey = DummyRoleARN
|
||||||
hasLegacyPolicyMapping = true
|
// Ensure that at most one JWT policy claim based provider may be
|
||||||
// Ensure that when a JWT policy claim based provider
|
// defined.
|
||||||
// exists, it is the only one.
|
|
||||||
if _, ok := c.arnProviderCfgsMap[DummyRoleARN]; ok {
|
if _, ok := c.arnProviderCfgsMap[DummyRoleARN]; ok {
|
||||||
return c, errSingleProvider
|
return c, errSingleProvider
|
||||||
}
|
}
|
||||||
@ -392,12 +388,6 @@ func LookupConfig(s config.Config, transport http.RoundTripper, closeRespFn func
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Ensure that when a JWT policy claim based provider
|
|
||||||
// exists, it is the only one.
|
|
||||||
if hasLegacyPolicyMapping && len(c.ProviderCfgs) > 1 {
|
|
||||||
return c, errSingleProvider
|
|
||||||
}
|
|
||||||
|
|
||||||
c.Enabled = true
|
c.Enabled = true
|
||||||
|
|
||||||
return c, nil
|
return c, nil
|
||||||
|
Loading…
Reference in New Issue
Block a user