mirror of
https://github.com/minio/minio.git
synced 2025-01-11 15:03:22 -05:00
Add IAM service account tests (#13502)
This commit is contained in:
parent
0c76fb57f2
commit
3b9dfa9d29
@ -92,6 +92,7 @@ func runAllIAMTests(suite *TestSuiteIAM, c *check) {
|
|||||||
suite.TestUserCreate(c)
|
suite.TestUserCreate(c)
|
||||||
suite.TestPolicyCreate(c)
|
suite.TestPolicyCreate(c)
|
||||||
suite.TestGroupAddRemove(c)
|
suite.TestGroupAddRemove(c)
|
||||||
|
suite.TestServiceAccountOps(c)
|
||||||
suite.TearDownSuite(c)
|
suite.TearDownSuite(c)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -232,25 +233,15 @@ func (s *TestSuiteIAM) TestPolicyCreate(c *check) {
|
|||||||
}
|
}
|
||||||
// 3.1 check that user does not have any access to the bucket
|
// 3.1 check that user does not have any access to the bucket
|
||||||
uClient := s.getUserClient(c, accessKey, secretKey, "")
|
uClient := s.getUserClient(c, accessKey, secretKey, "")
|
||||||
res := uClient.ListObjects(ctx, bucket, minio.ListObjectsOptions{})
|
c.mustNotListObjects(ctx, uClient, bucket)
|
||||||
v, ok := <-res
|
|
||||||
if !ok {
|
|
||||||
c.Fatalf("list channel was closed!")
|
|
||||||
}
|
|
||||||
if v.Err == nil {
|
|
||||||
c.Fatalf("User appears to be able to list!")
|
|
||||||
}
|
|
||||||
// 3.2 associate policy to user
|
// 3.2 associate policy to user
|
||||||
err = s.adm.SetPolicy(ctx, policy, accessKey, false)
|
err = s.adm.SetPolicy(ctx, policy, accessKey, false)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
c.Fatalf("Unable to set policy: %v", err)
|
c.Fatalf("Unable to set policy: %v", err)
|
||||||
}
|
}
|
||||||
// 3.3 check user has access to bucket
|
// 3.3 check user has access to bucket
|
||||||
res = uClient.ListObjects(ctx, bucket, minio.ListObjectsOptions{})
|
c.mustListObjects(ctx, uClient, bucket)
|
||||||
v, ok = <-res
|
|
||||||
if ok {
|
|
||||||
c.Fatalf("list channel was not closed - unexpected error or objects in listing: %v", v)
|
|
||||||
}
|
|
||||||
// 3.4 Check that user cannot exceed their permissions
|
// 3.4 Check that user cannot exceed their permissions
|
||||||
err = uClient.RemoveBucket(ctx, bucket)
|
err = uClient.RemoveBucket(ctx, bucket)
|
||||||
if err == nil {
|
if err == nil {
|
||||||
@ -262,7 +253,7 @@ func (s *TestSuiteIAM) TestPolicyCreate(c *check) {
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
c.Fatalf("policy list err: %v", err)
|
c.Fatalf("policy list err: %v", err)
|
||||||
}
|
}
|
||||||
_, ok = ps[policy]
|
_, ok := ps[policy]
|
||||||
if !ok {
|
if !ok {
|
||||||
c.Fatalf("policy was missing!")
|
c.Fatalf("policy was missing!")
|
||||||
}
|
}
|
||||||
@ -324,14 +315,7 @@ func (s *TestSuiteIAM) TestGroupAddRemove(c *check) {
|
|||||||
|
|
||||||
// 2. Check that user has no access
|
// 2. Check that user has no access
|
||||||
uClient := s.getUserClient(c, accessKey, secretKey, "")
|
uClient := s.getUserClient(c, accessKey, secretKey, "")
|
||||||
res := uClient.ListObjects(ctx, bucket, minio.ListObjectsOptions{})
|
c.mustNotListObjects(ctx, uClient, bucket)
|
||||||
v, ok := <-res
|
|
||||||
if !ok {
|
|
||||||
c.Fatalf("list channel was closed!")
|
|
||||||
}
|
|
||||||
if v.Err == nil {
|
|
||||||
c.Fatalf("User appears to be able to list!")
|
|
||||||
}
|
|
||||||
|
|
||||||
// 3. Associate policy to group and check user got access.
|
// 3. Associate policy to group and check user got access.
|
||||||
err = s.adm.SetPolicy(ctx, policy, group, true)
|
err = s.adm.SetPolicy(ctx, policy, group, true)
|
||||||
@ -339,11 +323,7 @@ func (s *TestSuiteIAM) TestGroupAddRemove(c *check) {
|
|||||||
c.Fatalf("Unable to set policy: %v", err)
|
c.Fatalf("Unable to set policy: %v", err)
|
||||||
}
|
}
|
||||||
// 3.1 check user has access to bucket
|
// 3.1 check user has access to bucket
|
||||||
res = uClient.ListObjects(ctx, bucket, minio.ListObjectsOptions{})
|
c.mustListObjects(ctx, uClient, bucket)
|
||||||
v, ok = <-res
|
|
||||||
if ok {
|
|
||||||
c.Fatalf("list channel was not closed - unexpected error or objects in listing: %v", v)
|
|
||||||
}
|
|
||||||
// 3.2 Check that user cannot exceed their permissions
|
// 3.2 Check that user cannot exceed their permissions
|
||||||
err = uClient.RemoveBucket(ctx, bucket)
|
err = uClient.RemoveBucket(ctx, bucket)
|
||||||
if err == nil {
|
if err == nil {
|
||||||
@ -377,14 +357,8 @@ func (s *TestSuiteIAM) TestGroupAddRemove(c *check) {
|
|||||||
c.Fatalf("group desc err: %v", err)
|
c.Fatalf("group desc err: %v", err)
|
||||||
}
|
}
|
||||||
c.Assert(groupInfo.Status, string(madmin.GroupDisabled))
|
c.Assert(groupInfo.Status, string(madmin.GroupDisabled))
|
||||||
res = uClient.ListObjects(ctx, bucket, minio.ListObjectsOptions{})
|
c.mustNotListObjects(ctx, uClient, bucket)
|
||||||
v, ok = <-res
|
|
||||||
if !ok {
|
|
||||||
c.Fatalf("list channel was closed!")
|
|
||||||
}
|
|
||||||
if v.Err == nil {
|
|
||||||
c.Fatalf("User appears to be able to list!")
|
|
||||||
}
|
|
||||||
err = s.adm.SetGroupStatus(ctx, group, madmin.GroupEnabled)
|
err = s.adm.SetGroupStatus(ctx, group, madmin.GroupEnabled)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
c.Fatalf("group set status err: %v", err)
|
c.Fatalf("group set status err: %v", err)
|
||||||
@ -394,11 +368,7 @@ func (s *TestSuiteIAM) TestGroupAddRemove(c *check) {
|
|||||||
c.Fatalf("group desc err: %v", err)
|
c.Fatalf("group desc err: %v", err)
|
||||||
}
|
}
|
||||||
c.Assert(groupInfo.Status, string(madmin.GroupEnabled))
|
c.Assert(groupInfo.Status, string(madmin.GroupEnabled))
|
||||||
res = uClient.ListObjects(ctx, bucket, minio.ListObjectsOptions{})
|
c.mustListObjects(ctx, uClient, bucket)
|
||||||
v, ok = <-res
|
|
||||||
if ok {
|
|
||||||
c.Fatalf("list channel was not closed - unexpected error or objects in listing: %v", v)
|
|
||||||
}
|
|
||||||
|
|
||||||
// 6. Verify that group cannot be deleted with users.
|
// 6. Verify that group cannot be deleted with users.
|
||||||
err = s.adm.UpdateGroupMembers(ctx, madmin.GroupAddRemove{
|
err = s.adm.UpdateGroupMembers(ctx, madmin.GroupAddRemove{
|
||||||
@ -423,14 +393,8 @@ func (s *TestSuiteIAM) TestGroupAddRemove(c *check) {
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
c.Fatalf("group update err: %v", err)
|
c.Fatalf("group update err: %v", err)
|
||||||
}
|
}
|
||||||
res = uClient.ListObjects(ctx, bucket, minio.ListObjectsOptions{})
|
c.mustNotListObjects(ctx, uClient, bucket)
|
||||||
v, ok = <-res
|
|
||||||
if !ok {
|
|
||||||
c.Fatalf("list channel was closed!")
|
|
||||||
}
|
|
||||||
if v.Err == nil {
|
|
||||||
c.Fatalf("User appears to be able to list!")
|
|
||||||
}
|
|
||||||
// 7.1 verify group still exists
|
// 7.1 verify group still exists
|
||||||
groupInfo, err = s.adm.GetGroupDescription(ctx, group)
|
groupInfo, err = s.adm.GetGroupDescription(ctx, group)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@ -460,6 +424,213 @@ func (s *TestSuiteIAM) TestGroupAddRemove(c *check) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (s *TestSuiteIAM) TestServiceAccountOps(c *check) {
|
||||||
|
ctx, cancel := context.WithTimeout(context.Background(), testDefaultTimeout)
|
||||||
|
defer cancel()
|
||||||
|
|
||||||
|
bucket := getRandomBucketName()
|
||||||
|
err := s.client.MakeBucket(ctx, bucket, minio.MakeBucketOptions{})
|
||||||
|
if err != nil {
|
||||||
|
c.Fatalf("bucket creat error: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Create policy, user and associate policy
|
||||||
|
policy := "mypolicy"
|
||||||
|
policyBytes := []byte(fmt.Sprintf(`{
|
||||||
|
"Version": "2012-10-17",
|
||||||
|
"Statement": [
|
||||||
|
{
|
||||||
|
"Effect": "Allow",
|
||||||
|
"Action": [
|
||||||
|
"s3:PutObject",
|
||||||
|
"s3:GetObject",
|
||||||
|
"s3:ListBucket"
|
||||||
|
],
|
||||||
|
"Resource": [
|
||||||
|
"arn:aws:s3:::%s/*"
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}`, bucket))
|
||||||
|
err = s.adm.AddCannedPolicy(ctx, policy, policyBytes)
|
||||||
|
if err != nil {
|
||||||
|
c.Fatalf("policy add error: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
accessKey, secretKey := mustGenerateCredentials(c)
|
||||||
|
err = s.adm.SetUser(ctx, accessKey, secretKey, madmin.AccountEnabled)
|
||||||
|
if err != nil {
|
||||||
|
c.Fatalf("Unable to set user: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
err = s.adm.SetPolicy(ctx, policy, accessKey, false)
|
||||||
|
if err != nil {
|
||||||
|
c.Fatalf("Unable to set policy: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// 1. Create a service account for the user
|
||||||
|
svcAK, svcSK := mustGenerateCredentials(c)
|
||||||
|
cr, err := s.adm.AddServiceAccount(ctx, madmin.AddServiceAccountReq{
|
||||||
|
TargetUser: accessKey,
|
||||||
|
AccessKey: svcAK,
|
||||||
|
SecretKey: svcSK,
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
c.Fatalf("Unable to create svc acc: %v", err)
|
||||||
|
}
|
||||||
|
// 1.2 Check that svc account appears in listing
|
||||||
|
listResp, err := s.adm.ListServiceAccounts(ctx, accessKey)
|
||||||
|
if err != nil {
|
||||||
|
c.Fatalf("unable to list svc accounts: %v", err)
|
||||||
|
}
|
||||||
|
if !set.CreateStringSet(listResp.Accounts...).Contains(svcAK) {
|
||||||
|
c.Fatalf("created service account did not appear in listing!")
|
||||||
|
}
|
||||||
|
// 1.3 Check that svc account info can be queried
|
||||||
|
infoResp, err := s.adm.InfoServiceAccount(ctx, svcAK)
|
||||||
|
if err != nil {
|
||||||
|
c.Fatalf("unable to get svc acc info: %v", err)
|
||||||
|
}
|
||||||
|
c.Assert(infoResp.ParentUser, accessKey)
|
||||||
|
c.Assert(infoResp.AccountStatus, "on")
|
||||||
|
c.Assert(infoResp.ImpliedPolicy, true)
|
||||||
|
|
||||||
|
// 2. Check that svc account can access the bucket
|
||||||
|
{
|
||||||
|
svcClient := s.getUserClient(c, cr.AccessKey, cr.SecretKey, "")
|
||||||
|
c.mustListObjects(ctx, svcClient, bucket)
|
||||||
|
}
|
||||||
|
|
||||||
|
// 3. Check that svc account can restrict the policy, and that the
|
||||||
|
// session policy can be updated.
|
||||||
|
{
|
||||||
|
svcAK, svcSK := mustGenerateCredentials(c)
|
||||||
|
policyBytes := []byte(fmt.Sprintf(`{
|
||||||
|
"Version": "2012-10-17",
|
||||||
|
"Statement": [
|
||||||
|
{
|
||||||
|
"Effect": "Allow",
|
||||||
|
"Action": [
|
||||||
|
"s3:PutObject",
|
||||||
|
"s3:GetObject"
|
||||||
|
],
|
||||||
|
"Resource": [
|
||||||
|
"arn:aws:s3:::%s/*"
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}`, bucket))
|
||||||
|
cr, err := s.adm.AddServiceAccount(ctx, madmin.AddServiceAccountReq{
|
||||||
|
Policy: policyBytes,
|
||||||
|
TargetUser: accessKey,
|
||||||
|
AccessKey: svcAK,
|
||||||
|
SecretKey: svcSK,
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
c.Fatalf("Unable to create svc acc: %v", err)
|
||||||
|
}
|
||||||
|
svcClient := s.getUserClient(c, cr.AccessKey, cr.SecretKey, "")
|
||||||
|
c.mustNotListObjects(ctx, svcClient, bucket)
|
||||||
|
|
||||||
|
newPolicyBytes := []byte(fmt.Sprintf(`{
|
||||||
|
"Version": "2012-10-17",
|
||||||
|
"Statement": [
|
||||||
|
{
|
||||||
|
"Effect": "Allow",
|
||||||
|
"Action": [
|
||||||
|
"s3:ListBucket"
|
||||||
|
],
|
||||||
|
"Resource": [
|
||||||
|
"arn:aws:s3:::%s/*"
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}`, bucket))
|
||||||
|
err = s.adm.UpdateServiceAccount(ctx, svcAK, madmin.UpdateServiceAccountReq{
|
||||||
|
NewPolicy: newPolicyBytes,
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
c.Fatalf("unable to update session policy for svc acc: %v", err)
|
||||||
|
}
|
||||||
|
c.mustListObjects(ctx, svcClient, bucket)
|
||||||
|
}
|
||||||
|
|
||||||
|
// 4. Check that service account's secret key and account status can be
|
||||||
|
// updated.
|
||||||
|
{
|
||||||
|
svcAK, svcSK := mustGenerateCredentials(c)
|
||||||
|
cr, err := s.adm.AddServiceAccount(ctx, madmin.AddServiceAccountReq{
|
||||||
|
TargetUser: accessKey,
|
||||||
|
AccessKey: svcAK,
|
||||||
|
SecretKey: svcSK,
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
c.Fatalf("Unable to create svc acc: %v", err)
|
||||||
|
}
|
||||||
|
svcClient := s.getUserClient(c, cr.AccessKey, cr.SecretKey, "")
|
||||||
|
c.mustListObjects(ctx, svcClient, bucket)
|
||||||
|
|
||||||
|
_, svcSK2 := mustGenerateCredentials(c)
|
||||||
|
err = s.adm.UpdateServiceAccount(ctx, svcAK, madmin.UpdateServiceAccountReq{
|
||||||
|
NewSecretKey: svcSK2,
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
c.Fatalf("unable to update secret key for svc acc: %v", err)
|
||||||
|
}
|
||||||
|
// old creds should not work:
|
||||||
|
c.mustNotListObjects(ctx, svcClient, bucket)
|
||||||
|
// new creds work:
|
||||||
|
svcClient2 := s.getUserClient(c, cr.AccessKey, svcSK2, "")
|
||||||
|
c.mustListObjects(ctx, svcClient2, bucket)
|
||||||
|
|
||||||
|
// update status to disabled
|
||||||
|
err = s.adm.UpdateServiceAccount(ctx, svcAK, madmin.UpdateServiceAccountReq{
|
||||||
|
NewStatus: "off",
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
c.Fatalf("unable to update secret key for svc acc: %v", err)
|
||||||
|
}
|
||||||
|
c.mustNotListObjects(ctx, svcClient2, bucket)
|
||||||
|
}
|
||||||
|
|
||||||
|
// 5. Check that service account can be deleted.
|
||||||
|
{
|
||||||
|
svcAK, svcSK := mustGenerateCredentials(c)
|
||||||
|
cr, err := s.adm.AddServiceAccount(ctx, madmin.AddServiceAccountReq{
|
||||||
|
TargetUser: accessKey,
|
||||||
|
AccessKey: svcAK,
|
||||||
|
SecretKey: svcSK,
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
c.Fatalf("Unable to create svc acc: %v", err)
|
||||||
|
}
|
||||||
|
svcClient := s.getUserClient(c, cr.AccessKey, cr.SecretKey, "")
|
||||||
|
c.mustListObjects(ctx, svcClient, bucket)
|
||||||
|
|
||||||
|
err = s.adm.DeleteServiceAccount(ctx, svcAK)
|
||||||
|
if err != nil {
|
||||||
|
c.Fatalf("unable to delete svc acc: %v", err)
|
||||||
|
}
|
||||||
|
c.mustNotListObjects(ctx, svcClient, bucket)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *check) mustNotListObjects(ctx context.Context, client *minio.Client, bucket string) {
|
||||||
|
res := client.ListObjects(ctx, bucket, minio.ListObjectsOptions{})
|
||||||
|
v, ok := <-res
|
||||||
|
if !ok || v.Err == nil {
|
||||||
|
c.Fatalf("user was able to list unexpectedly!")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *check) mustListObjects(ctx context.Context, client *minio.Client, bucket string) {
|
||||||
|
res := client.ListObjects(ctx, bucket, minio.ListObjectsOptions{})
|
||||||
|
v, ok := <-res
|
||||||
|
if ok && v.Err != nil {
|
||||||
|
c.Fatalf("user was unable to list unexpectedly!")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
func mustGenerateCredentials(c *check) (string, string) {
|
func mustGenerateCredentials(c *check) (string, string) {
|
||||||
ak, sk, err := auth.GenerateCredentials()
|
ak, sk, err := auth.GenerateCredentials()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
Loading…
Reference in New Issue
Block a user