mirror of https://github.com/minio/minio.git
852 lines
22 KiB
Go
852 lines
22 KiB
Go
|
// Copyright (c) 2015-2024 MinIO, Inc.
|
||
|
//
|
||
|
// This file is part of MinIO Object Storage stack
|
||
|
//
|
||
|
// This program is free software: you can redistribute it and/or modify
|
||
|
// it under the terms of the GNU Affero General Public License as published by
|
||
|
// the Free Software Foundation, either version 3 of the License, or
|
||
|
// (at your option) any later version.
|
||
|
//
|
||
|
// This program is distributed in the hope that it will be useful
|
||
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||
|
// GNU Affero General Public License for more details.
|
||
|
//
|
||
|
// You should have received a copy of the GNU Affero General Public License
|
||
|
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||
|
|
||
|
package cmd
|
||
|
|
||
|
import (
|
||
|
"context"
|
||
|
"encoding/json"
|
||
|
"fmt"
|
||
|
"net/http"
|
||
|
"net/http/httptest"
|
||
|
"net/url"
|
||
|
"strings"
|
||
|
"testing"
|
||
|
|
||
|
"github.com/minio/madmin-go/v3"
|
||
|
"github.com/minio/minio/internal/kms"
|
||
|
"github.com/minio/pkg/v3/policy"
|
||
|
)
|
||
|
|
||
|
const (
|
||
|
// KMS API paths
|
||
|
// For example: /minio/kms/v1/key/list?pattern=*
|
||
|
kmsURL = kmsPathPrefix + kmsAPIVersionPrefix
|
||
|
kmsStatusPath = kmsURL + "/status"
|
||
|
kmsMetricsPath = kmsURL + "/metrics"
|
||
|
kmsAPIsPath = kmsURL + "/apis"
|
||
|
kmsVersionPath = kmsURL + "/version"
|
||
|
kmsKeyCreatePath = kmsURL + "/key/create"
|
||
|
kmsKeyListPath = kmsURL + "/key/list"
|
||
|
kmsKeyStatusPath = kmsURL + "/key/status"
|
||
|
|
||
|
// Admin API paths
|
||
|
// For example: /minio/admin/v3/kms/status
|
||
|
adminURL = adminPathPrefix + adminAPIVersionPrefix
|
||
|
kmsAdminStatusPath = adminURL + "/kms/status"
|
||
|
kmsAdminKeyStatusPath = adminURL + "/kms/key/status"
|
||
|
kmsAdminKeyCreate = adminURL + "/kms/key/create"
|
||
|
)
|
||
|
|
||
|
const (
|
||
|
userAccessKey = "miniofakeuseraccesskey"
|
||
|
userSecretKey = "miniofakeusersecret"
|
||
|
)
|
||
|
|
||
|
type kmsTestCase struct {
|
||
|
name string
|
||
|
method string
|
||
|
path string
|
||
|
query map[string]string
|
||
|
|
||
|
// User credentials and policy for request
|
||
|
policy string
|
||
|
asRoot bool
|
||
|
|
||
|
// Wanted in response.
|
||
|
wantStatusCode int
|
||
|
wantKeyNames []string
|
||
|
wantResp []string
|
||
|
}
|
||
|
|
||
|
func TestKMSHandlersCreateKey(t *testing.T) {
|
||
|
adminTestBed, tearDown := setupKMSTest(t, true)
|
||
|
defer tearDown()
|
||
|
|
||
|
tests := []kmsTestCase{
|
||
|
// Create key test
|
||
|
{
|
||
|
name: "create key as user with no policy want forbidden",
|
||
|
method: http.MethodPost,
|
||
|
path: kmsKeyCreatePath,
|
||
|
query: map[string]string{"key-id": "new-test-key"},
|
||
|
asRoot: false,
|
||
|
|
||
|
wantStatusCode: http.StatusForbidden,
|
||
|
wantResp: []string{"AccessDenied"},
|
||
|
},
|
||
|
{
|
||
|
name: "create key as user with no resources specified want success",
|
||
|
method: http.MethodPost,
|
||
|
path: kmsKeyCreatePath,
|
||
|
query: map[string]string{"key-id": "new-test-key"},
|
||
|
asRoot: false,
|
||
|
|
||
|
policy: `{"Effect": "Allow",
|
||
|
"Action": ["kms:CreateKey"] }`,
|
||
|
|
||
|
wantStatusCode: http.StatusOK,
|
||
|
},
|
||
|
{
|
||
|
name: "create key as user set policy to allow want success",
|
||
|
method: http.MethodPost,
|
||
|
path: kmsKeyCreatePath,
|
||
|
query: map[string]string{"key-id": "second-new-test-key"},
|
||
|
asRoot: false,
|
||
|
|
||
|
policy: `{"Effect": "Allow",
|
||
|
"Action": ["kms:CreateKey"],
|
||
|
"Resource": ["arn:minio:kms:::second-new-test-*"] }`,
|
||
|
|
||
|
wantStatusCode: http.StatusOK,
|
||
|
},
|
||
|
{
|
||
|
name: "create key as user set policy to non matching resource want forbidden",
|
||
|
method: http.MethodPost,
|
||
|
path: kmsKeyCreatePath,
|
||
|
query: map[string]string{"key-id": "third-new-test-key"},
|
||
|
asRoot: false,
|
||
|
|
||
|
policy: `{"Effect": "Allow",
|
||
|
"Action": ["kms:CreateKey"],
|
||
|
"Resource": ["arn:minio:kms:::non-matching-key-name"] }`,
|
||
|
|
||
|
wantStatusCode: http.StatusForbidden,
|
||
|
wantResp: []string{"AccessDenied"},
|
||
|
},
|
||
|
}
|
||
|
for testNum, test := range tests {
|
||
|
t.Run(fmt.Sprintf("%d %s", testNum+1, test.name), func(t *testing.T) {
|
||
|
execKMSTest(t, test, adminTestBed)
|
||
|
})
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func TestKMSHandlersKeyStatus(t *testing.T) {
|
||
|
adminTestBed, tearDown := setupKMSTest(t, true)
|
||
|
defer tearDown()
|
||
|
|
||
|
tests := []kmsTestCase{
|
||
|
{
|
||
|
name: "create a first key root user",
|
||
|
method: http.MethodPost,
|
||
|
path: kmsKeyCreatePath,
|
||
|
query: map[string]string{"key-id": "abc-test-key"},
|
||
|
asRoot: true,
|
||
|
|
||
|
wantStatusCode: http.StatusOK,
|
||
|
},
|
||
|
{
|
||
|
name: "key status as root want success",
|
||
|
method: http.MethodGet,
|
||
|
path: kmsKeyStatusPath,
|
||
|
query: map[string]string{"key-id": "abc-test-key"},
|
||
|
asRoot: true,
|
||
|
|
||
|
wantStatusCode: http.StatusOK,
|
||
|
wantResp: []string{"abc-test-key"},
|
||
|
},
|
||
|
{
|
||
|
name: "key status as user no policy want forbidden",
|
||
|
method: http.MethodGet,
|
||
|
path: kmsKeyStatusPath,
|
||
|
query: map[string]string{"key-id": "abc-test-key"},
|
||
|
asRoot: false,
|
||
|
|
||
|
wantStatusCode: http.StatusForbidden,
|
||
|
wantResp: []string{"AccessDenied"},
|
||
|
},
|
||
|
{
|
||
|
name: "key status as user legacy no resources specified want success",
|
||
|
method: http.MethodGet,
|
||
|
path: kmsKeyStatusPath,
|
||
|
query: map[string]string{"key-id": "abc-test-key"},
|
||
|
asRoot: false,
|
||
|
|
||
|
policy: `{"Effect": "Allow",
|
||
|
"Action": ["kms:KeyStatus"] }`,
|
||
|
|
||
|
wantStatusCode: http.StatusOK,
|
||
|
wantResp: []string{"abc-test-key"},
|
||
|
},
|
||
|
{
|
||
|
name: "key status as user set policy to allow only one key",
|
||
|
method: http.MethodGet,
|
||
|
path: kmsKeyStatusPath,
|
||
|
query: map[string]string{"key-id": "abc-test-key"},
|
||
|
asRoot: false,
|
||
|
|
||
|
policy: `{"Effect": "Allow",
|
||
|
"Action": ["kms:KeyStatus"],
|
||
|
"Resource": ["arn:minio:kms:::abc-test-*"] }`,
|
||
|
|
||
|
wantStatusCode: http.StatusOK,
|
||
|
wantResp: []string{"abc-test-key"},
|
||
|
},
|
||
|
{
|
||
|
name: "key status as user set policy to allow non-matching key",
|
||
|
method: http.MethodGet,
|
||
|
path: kmsKeyStatusPath,
|
||
|
query: map[string]string{"key-id": "abc-test-key"},
|
||
|
asRoot: false,
|
||
|
|
||
|
policy: `{"Effect": "Allow",
|
||
|
"Action": ["kms:KeyStatus"],
|
||
|
"Resource": ["arn:minio:kms:::xyz-test-key"] }`,
|
||
|
|
||
|
wantStatusCode: http.StatusForbidden,
|
||
|
wantResp: []string{"AccessDenied"},
|
||
|
},
|
||
|
}
|
||
|
for testNum, test := range tests {
|
||
|
t.Run(fmt.Sprintf("%d %s", testNum+1, test.name), func(t *testing.T) {
|
||
|
execKMSTest(t, test, adminTestBed)
|
||
|
})
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func TestKMSHandlersAPIs(t *testing.T) {
|
||
|
adminTestBed, tearDown := setupKMSTest(t, true)
|
||
|
defer tearDown()
|
||
|
|
||
|
tests := []kmsTestCase{
|
||
|
// Version test
|
||
|
{
|
||
|
name: "version as root want success",
|
||
|
method: http.MethodGet,
|
||
|
path: kmsVersionPath,
|
||
|
asRoot: true,
|
||
|
|
||
|
wantStatusCode: http.StatusOK,
|
||
|
wantResp: []string{"version"},
|
||
|
},
|
||
|
{
|
||
|
name: "version as user with no policy want forbidden",
|
||
|
method: http.MethodGet,
|
||
|
path: kmsVersionPath,
|
||
|
asRoot: false,
|
||
|
|
||
|
wantStatusCode: http.StatusForbidden,
|
||
|
wantResp: []string{"AccessDenied"},
|
||
|
},
|
||
|
{
|
||
|
name: "version as user with policy ignores resource want success",
|
||
|
method: http.MethodGet,
|
||
|
path: kmsVersionPath,
|
||
|
asRoot: false,
|
||
|
|
||
|
policy: `{"Effect": "Allow",
|
||
|
"Action": ["kms:Version"],
|
||
|
"Resource": ["arn:minio:kms:::does-not-matter-it-is-ignored"] }`,
|
||
|
|
||
|
wantStatusCode: http.StatusOK,
|
||
|
wantResp: []string{"version"},
|
||
|
},
|
||
|
|
||
|
// APIs test
|
||
|
{
|
||
|
name: "apis as root want success",
|
||
|
method: http.MethodGet,
|
||
|
path: kmsAPIsPath,
|
||
|
asRoot: true,
|
||
|
|
||
|
wantStatusCode: http.StatusOK,
|
||
|
wantResp: []string{"stub/path"},
|
||
|
},
|
||
|
{
|
||
|
name: "apis as user with no policy want forbidden",
|
||
|
method: http.MethodGet,
|
||
|
path: kmsAPIsPath,
|
||
|
asRoot: false,
|
||
|
|
||
|
wantStatusCode: http.StatusForbidden,
|
||
|
wantResp: []string{"AccessDenied"},
|
||
|
},
|
||
|
{
|
||
|
name: "apis as user with policy ignores resource want success",
|
||
|
method: http.MethodGet,
|
||
|
path: kmsAPIsPath,
|
||
|
asRoot: false,
|
||
|
|
||
|
policy: `{"Effect": "Allow",
|
||
|
"Action": ["kms:API"],
|
||
|
"Resource": ["arn:minio:kms:::does-not-matter-it-is-ignored"] }`,
|
||
|
|
||
|
wantStatusCode: http.StatusOK,
|
||
|
wantResp: []string{"stub/path"},
|
||
|
},
|
||
|
|
||
|
// Metrics test
|
||
|
{
|
||
|
name: "metrics as root want success",
|
||
|
method: http.MethodGet,
|
||
|
path: kmsMetricsPath,
|
||
|
asRoot: true,
|
||
|
|
||
|
wantStatusCode: http.StatusOK,
|
||
|
wantResp: []string{"kms"},
|
||
|
},
|
||
|
{
|
||
|
name: "metrics as user with no policy want forbidden",
|
||
|
method: http.MethodGet,
|
||
|
path: kmsMetricsPath,
|
||
|
asRoot: false,
|
||
|
|
||
|
wantStatusCode: http.StatusForbidden,
|
||
|
wantResp: []string{"AccessDenied"},
|
||
|
},
|
||
|
{
|
||
|
name: "metrics as user with policy ignores resource want success",
|
||
|
method: http.MethodGet,
|
||
|
path: kmsMetricsPath,
|
||
|
asRoot: false,
|
||
|
|
||
|
policy: `{"Effect": "Allow",
|
||
|
"Action": ["kms:Metrics"],
|
||
|
"Resource": ["arn:minio:kms:::does-not-matter-it-is-ignored"] }`,
|
||
|
|
||
|
wantStatusCode: http.StatusOK,
|
||
|
wantResp: []string{"kms"},
|
||
|
},
|
||
|
|
||
|
// Status tests
|
||
|
{
|
||
|
name: "status as root want success",
|
||
|
method: http.MethodGet,
|
||
|
path: kmsStatusPath,
|
||
|
asRoot: true,
|
||
|
|
||
|
wantStatusCode: http.StatusOK,
|
||
|
wantResp: []string{"MinIO builtin"},
|
||
|
},
|
||
|
{
|
||
|
name: "status as user with no policy want forbidden",
|
||
|
method: http.MethodGet,
|
||
|
path: kmsStatusPath,
|
||
|
asRoot: false,
|
||
|
|
||
|
wantStatusCode: http.StatusForbidden,
|
||
|
wantResp: []string{"AccessDenied"},
|
||
|
},
|
||
|
{
|
||
|
name: "status as user with policy ignores resource want success",
|
||
|
method: http.MethodGet,
|
||
|
path: kmsStatusPath,
|
||
|
asRoot: false,
|
||
|
|
||
|
policy: `{"Effect": "Allow",
|
||
|
"Action": ["kms:Status"],
|
||
|
"Resource": ["arn:minio:kms:::does-not-matter-it-is-ignored"]}`,
|
||
|
|
||
|
wantStatusCode: http.StatusOK,
|
||
|
wantResp: []string{"MinIO builtin"},
|
||
|
},
|
||
|
}
|
||
|
for testNum, test := range tests {
|
||
|
t.Run(fmt.Sprintf("%d %s", testNum+1, test.name), func(t *testing.T) {
|
||
|
execKMSTest(t, test, adminTestBed)
|
||
|
})
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func TestKMSHandlersListKeys(t *testing.T) {
|
||
|
adminTestBed, tearDown := setupKMSTest(t, true)
|
||
|
defer tearDown()
|
||
|
|
||
|
tests := []kmsTestCase{
|
||
|
{
|
||
|
name: "create a first key root user",
|
||
|
method: http.MethodPost,
|
||
|
path: kmsKeyCreatePath,
|
||
|
query: map[string]string{"key-id": "abc-test-key"},
|
||
|
asRoot: true,
|
||
|
|
||
|
wantStatusCode: http.StatusOK,
|
||
|
},
|
||
|
{
|
||
|
name: "create a second key root user",
|
||
|
method: http.MethodPost,
|
||
|
path: kmsKeyCreatePath,
|
||
|
query: map[string]string{"key-id": "xyz-test-key"},
|
||
|
asRoot: true,
|
||
|
|
||
|
wantStatusCode: http.StatusOK,
|
||
|
},
|
||
|
|
||
|
// List keys tests
|
||
|
{
|
||
|
name: "list keys as root want all to be returned",
|
||
|
method: http.MethodGet,
|
||
|
path: kmsKeyListPath,
|
||
|
query: map[string]string{"pattern": "*"},
|
||
|
asRoot: true,
|
||
|
|
||
|
wantStatusCode: http.StatusOK,
|
||
|
wantKeyNames: []string{"default-test-key", "abc-test-key", "xyz-test-key"},
|
||
|
},
|
||
|
{
|
||
|
name: "list keys as user with no policy want forbidden",
|
||
|
method: http.MethodGet,
|
||
|
path: kmsKeyListPath,
|
||
|
query: map[string]string{"pattern": "*"},
|
||
|
asRoot: false,
|
||
|
|
||
|
wantStatusCode: http.StatusForbidden,
|
||
|
wantResp: []string{"AccessDenied"},
|
||
|
},
|
||
|
{
|
||
|
name: "list keys as user with no resources specified want success",
|
||
|
method: http.MethodGet,
|
||
|
path: kmsKeyListPath,
|
||
|
query: map[string]string{"pattern": "*"},
|
||
|
asRoot: false,
|
||
|
|
||
|
policy: `{"Effect": "Allow",
|
||
|
"Action": ["kms:ListKeys"]
|
||
|
}`,
|
||
|
|
||
|
wantStatusCode: http.StatusOK,
|
||
|
wantKeyNames: []string{"default-test-key", "abc-test-key", "xyz-test-key"},
|
||
|
},
|
||
|
{
|
||
|
name: "list keys as user set policy resource to allow only one key",
|
||
|
method: http.MethodGet,
|
||
|
path: kmsKeyListPath,
|
||
|
query: map[string]string{"pattern": "*"},
|
||
|
asRoot: false,
|
||
|
|
||
|
policy: `{"Effect": "Allow",
|
||
|
"Action": ["kms:ListKeys"],
|
||
|
"Resource": ["arn:minio:kms:::abc*"]}`,
|
||
|
|
||
|
wantStatusCode: http.StatusOK,
|
||
|
wantKeyNames: []string{"abc-test-key"},
|
||
|
},
|
||
|
{
|
||
|
name: "list keys as user set policy to allow only one key, use pattern that includes correct key",
|
||
|
method: http.MethodGet,
|
||
|
path: kmsKeyListPath,
|
||
|
query: map[string]string{"pattern": "abc*"},
|
||
|
|
||
|
policy: `{"Effect": "Allow",
|
||
|
"Action": ["kms:ListKeys"],
|
||
|
"Resource": ["arn:minio:kms:::abc*"]}`,
|
||
|
|
||
|
wantStatusCode: http.StatusOK,
|
||
|
wantKeyNames: []string{"abc-test-key"},
|
||
|
},
|
||
|
{
|
||
|
name: "list keys as user set policy to allow only one key, use pattern that excludes correct key",
|
||
|
method: http.MethodGet,
|
||
|
path: kmsKeyListPath,
|
||
|
query: map[string]string{"pattern": "xyz*"},
|
||
|
asRoot: false,
|
||
|
|
||
|
policy: `{"Effect": "Allow",
|
||
|
"Action": ["kms:ListKeys"],
|
||
|
"Resource": ["arn:minio:kms:::abc*"]}`,
|
||
|
|
||
|
wantStatusCode: http.StatusOK,
|
||
|
wantKeyNames: []string{},
|
||
|
},
|
||
|
{
|
||
|
name: "list keys as user set policy that has no matching key resources",
|
||
|
method: http.MethodGet,
|
||
|
path: kmsKeyListPath,
|
||
|
query: map[string]string{"pattern": "*"},
|
||
|
asRoot: false,
|
||
|
|
||
|
policy: `{"Effect": "Allow",
|
||
|
"Action": ["kms:ListKeys"],
|
||
|
"Resource": ["arn:minio:kms:::nonematch*"]}`,
|
||
|
|
||
|
wantStatusCode: http.StatusOK,
|
||
|
wantKeyNames: []string{},
|
||
|
},
|
||
|
{
|
||
|
name: "list keys as user set policy that allows listing but denies specific keys",
|
||
|
method: http.MethodGet,
|
||
|
path: kmsKeyListPath,
|
||
|
query: map[string]string{"pattern": "*"},
|
||
|
asRoot: false,
|
||
|
|
||
|
// It looks like this should allow listing any key that isn't "default-test-key", however
|
||
|
// the policy engine matches all Deny statements first, without regard to Resources (for KMS).
|
||
|
// This is for backwards compatibility where historically KMS statements ignored Resources.
|
||
|
policy: `{
|
||
|
"Effect": "Allow",
|
||
|
"Action": ["kms:ListKeys"]
|
||
|
},{
|
||
|
"Effect": "Deny",
|
||
|
"Action": ["kms:ListKeys"],
|
||
|
"Resource": ["arn:minio:kms:::default-test-key"]
|
||
|
}`,
|
||
|
|
||
|
wantStatusCode: http.StatusForbidden,
|
||
|
wantResp: []string{"AccessDenied"},
|
||
|
},
|
||
|
}
|
||
|
|
||
|
for testNum, test := range tests {
|
||
|
t.Run(fmt.Sprintf("%d %s", testNum+1, test.name), func(t *testing.T) {
|
||
|
execKMSTest(t, test, adminTestBed)
|
||
|
})
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func TestKMSHandlerAdminAPI(t *testing.T) {
|
||
|
adminTestBed, tearDown := setupKMSTest(t, true)
|
||
|
defer tearDown()
|
||
|
|
||
|
tests := []kmsTestCase{
|
||
|
// Create key tests
|
||
|
{
|
||
|
name: "create a key root user",
|
||
|
method: http.MethodPost,
|
||
|
path: kmsAdminKeyCreate,
|
||
|
query: map[string]string{"key-id": "abc-test-key"},
|
||
|
asRoot: true,
|
||
|
|
||
|
wantStatusCode: http.StatusOK,
|
||
|
},
|
||
|
{
|
||
|
name: "create key as user with no policy want forbidden",
|
||
|
method: http.MethodPost,
|
||
|
path: kmsAdminKeyCreate,
|
||
|
query: map[string]string{"key-id": "new-test-key"},
|
||
|
asRoot: false,
|
||
|
|
||
|
wantStatusCode: http.StatusForbidden,
|
||
|
wantResp: []string{"AccessDenied"},
|
||
|
},
|
||
|
{
|
||
|
name: "create key as user with no resources specified want success",
|
||
|
method: http.MethodPost,
|
||
|
path: kmsAdminKeyCreate,
|
||
|
query: map[string]string{"key-id": "new-test-key"},
|
||
|
asRoot: false,
|
||
|
|
||
|
policy: `{"Effect": "Allow",
|
||
|
"Action": ["admin:KMSCreateKey"] }`,
|
||
|
|
||
|
wantStatusCode: http.StatusOK,
|
||
|
},
|
||
|
{
|
||
|
name: "create key as user set policy to non matching resource want success",
|
||
|
method: http.MethodPost,
|
||
|
path: kmsAdminKeyCreate,
|
||
|
query: map[string]string{"key-id": "third-new-test-key"},
|
||
|
asRoot: false,
|
||
|
|
||
|
// Admin actions ignore Resources
|
||
|
policy: `{"Effect": "Allow",
|
||
|
"Action": ["admin:KMSCreateKey"],
|
||
|
"Resource": ["arn:minio:kms:::this-is-disregarded"] }`,
|
||
|
|
||
|
wantStatusCode: http.StatusOK,
|
||
|
},
|
||
|
|
||
|
// Status tests
|
||
|
{
|
||
|
name: "status as root want success",
|
||
|
method: http.MethodPost,
|
||
|
path: kmsAdminStatusPath,
|
||
|
asRoot: true,
|
||
|
|
||
|
wantStatusCode: http.StatusOK,
|
||
|
wantResp: []string{"MinIO builtin"},
|
||
|
},
|
||
|
{
|
||
|
name: "status as user with no policy want forbidden",
|
||
|
method: http.MethodPost,
|
||
|
path: kmsAdminStatusPath,
|
||
|
asRoot: false,
|
||
|
|
||
|
wantStatusCode: http.StatusForbidden,
|
||
|
wantResp: []string{"AccessDenied"},
|
||
|
},
|
||
|
{
|
||
|
name: "status as user with policy ignores resource want success",
|
||
|
method: http.MethodPost,
|
||
|
path: kmsAdminStatusPath,
|
||
|
asRoot: false,
|
||
|
|
||
|
policy: `{"Effect": "Allow",
|
||
|
"Action": ["admin:KMSKeyStatus"],
|
||
|
"Resource": ["arn:minio:kms:::does-not-matter-it-is-ignored"] }`,
|
||
|
|
||
|
wantStatusCode: http.StatusOK,
|
||
|
wantResp: []string{"MinIO builtin"},
|
||
|
},
|
||
|
|
||
|
// Key status tests
|
||
|
{
|
||
|
name: "key status as root want success",
|
||
|
method: http.MethodGet,
|
||
|
path: kmsAdminKeyStatusPath,
|
||
|
asRoot: true,
|
||
|
|
||
|
wantStatusCode: http.StatusOK,
|
||
|
wantResp: []string{"key-id"},
|
||
|
},
|
||
|
{
|
||
|
name: "key status as user with no policy want forbidden",
|
||
|
method: http.MethodGet,
|
||
|
path: kmsAdminKeyStatusPath,
|
||
|
asRoot: false,
|
||
|
|
||
|
wantStatusCode: http.StatusForbidden,
|
||
|
wantResp: []string{"AccessDenied"},
|
||
|
},
|
||
|
{
|
||
|
name: "key status as user with policy ignores resource want success",
|
||
|
method: http.MethodGet,
|
||
|
path: kmsAdminKeyStatusPath,
|
||
|
asRoot: false,
|
||
|
|
||
|
policy: `{"Effect": "Allow",
|
||
|
"Action": ["admin:KMSKeyStatus"],
|
||
|
"Resource": ["arn:minio:kms:::does-not-matter-it-is-ignored"] }`,
|
||
|
|
||
|
wantStatusCode: http.StatusOK,
|
||
|
wantResp: []string{"key-id"},
|
||
|
},
|
||
|
}
|
||
|
|
||
|
for testNum, test := range tests {
|
||
|
t.Run(fmt.Sprintf("%d %s", testNum+1, test.name), func(t *testing.T) {
|
||
|
execKMSTest(t, test, adminTestBed)
|
||
|
})
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// execKMSTest runs a single test case for KMS handlers
|
||
|
func execKMSTest(t *testing.T, test kmsTestCase, adminTestBed *adminErasureTestBed) {
|
||
|
var accessKey, secretKey string
|
||
|
if test.asRoot {
|
||
|
accessKey, secretKey = globalActiveCred.AccessKey, globalActiveCred.SecretKey
|
||
|
} else {
|
||
|
setupKMSUser(t, userAccessKey, userSecretKey, test.policy)
|
||
|
accessKey = userAccessKey
|
||
|
secretKey = userSecretKey
|
||
|
}
|
||
|
|
||
|
req := buildKMSRequest(t, test.method, test.path, accessKey, secretKey, test.query)
|
||
|
rec := httptest.NewRecorder()
|
||
|
adminTestBed.router.ServeHTTP(rec, req)
|
||
|
|
||
|
t.Logf("HTTP req: %s, resp code: %d, resp body: %s", req.URL.String(), rec.Code, rec.Body.String())
|
||
|
|
||
|
// Check status code
|
||
|
if rec.Code != test.wantStatusCode {
|
||
|
t.Errorf("want status code %d, got %d", test.wantStatusCode, rec.Code)
|
||
|
}
|
||
|
|
||
|
// Check returned key list is correct
|
||
|
if test.wantKeyNames != nil {
|
||
|
gotKeyNames := keyNamesFromListKeysResp(t, rec.Body.Bytes())
|
||
|
if len(test.wantKeyNames) != len(gotKeyNames) {
|
||
|
t.Fatalf("want keys len: %d, got len: %d", len(test.wantKeyNames), len(gotKeyNames))
|
||
|
}
|
||
|
for i, wantKeyName := range test.wantKeyNames {
|
||
|
if gotKeyNames[i] != wantKeyName {
|
||
|
t.Fatalf("want key name %s, in position %d, got %s", wantKeyName, i, gotKeyNames[i])
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Check generic text in the response
|
||
|
if test.wantResp != nil {
|
||
|
for _, want := range test.wantResp {
|
||
|
if !strings.Contains(rec.Body.String(), want) {
|
||
|
t.Fatalf("want response to contain %s, got %s", want, rec.Body.String())
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// TestKMSHandlerNotConfiguredOrInvalidCreds tests KMS handlers for situations where KMS is not configured
|
||
|
// or invalid credentials are provided.
|
||
|
func TestKMSHandlerNotConfiguredOrInvalidCreds(t *testing.T) {
|
||
|
adminTestBed, tearDown := setupKMSTest(t, false)
|
||
|
defer tearDown()
|
||
|
|
||
|
tests := []struct {
|
||
|
name string
|
||
|
method string
|
||
|
path string
|
||
|
query map[string]string
|
||
|
}{
|
||
|
{
|
||
|
name: "GET status",
|
||
|
method: http.MethodGet,
|
||
|
path: kmsStatusPath,
|
||
|
},
|
||
|
{
|
||
|
name: "GET metrics",
|
||
|
method: http.MethodGet,
|
||
|
path: kmsMetricsPath,
|
||
|
},
|
||
|
{
|
||
|
name: "GET apis",
|
||
|
method: http.MethodGet,
|
||
|
path: kmsAPIsPath,
|
||
|
},
|
||
|
{
|
||
|
name: "GET version",
|
||
|
method: http.MethodGet,
|
||
|
path: kmsVersionPath,
|
||
|
},
|
||
|
{
|
||
|
name: "POST key create",
|
||
|
method: http.MethodPost,
|
||
|
path: kmsKeyCreatePath,
|
||
|
query: map[string]string{"key-id": "master-key-id"},
|
||
|
},
|
||
|
{
|
||
|
name: "GET key list",
|
||
|
method: http.MethodGet,
|
||
|
path: kmsKeyListPath,
|
||
|
query: map[string]string{"pattern": "*"},
|
||
|
},
|
||
|
{
|
||
|
name: "GET key status",
|
||
|
method: http.MethodGet,
|
||
|
path: kmsKeyStatusPath,
|
||
|
query: map[string]string{"key-id": "master-key-id"},
|
||
|
},
|
||
|
}
|
||
|
|
||
|
// Test when the GlobalKMS is not configured
|
||
|
for _, test := range tests {
|
||
|
t.Run(test.name+" not configured", func(t *testing.T) {
|
||
|
req := buildKMSRequest(t, test.method, test.path, "", "", test.query)
|
||
|
rec := httptest.NewRecorder()
|
||
|
adminTestBed.router.ServeHTTP(rec, req)
|
||
|
if rec.Code != http.StatusNotImplemented {
|
||
|
t.Errorf("want status code %d, got %d", http.StatusNotImplemented, rec.Code)
|
||
|
}
|
||
|
})
|
||
|
}
|
||
|
|
||
|
// Test when the GlobalKMS is configured but the credentials are invalid
|
||
|
GlobalKMS = kms.NewStub("default-test-key")
|
||
|
for _, test := range tests {
|
||
|
t.Run(test.name+" invalid credentials", func(t *testing.T) {
|
||
|
req := buildKMSRequest(t, test.method, test.path, userAccessKey, userSecretKey, test.query)
|
||
|
rec := httptest.NewRecorder()
|
||
|
adminTestBed.router.ServeHTTP(rec, req)
|
||
|
if rec.Code != http.StatusForbidden {
|
||
|
t.Errorf("want status code %d, got %d", http.StatusForbidden, rec.Code)
|
||
|
}
|
||
|
})
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func setupKMSTest(t *testing.T, enableKMS bool) (*adminErasureTestBed, func()) {
|
||
|
adminTestBed, err := prepareAdminErasureTestBed(context.Background())
|
||
|
if err != nil {
|
||
|
t.Fatal(err)
|
||
|
}
|
||
|
registerKMSRouter(adminTestBed.router)
|
||
|
|
||
|
if enableKMS {
|
||
|
GlobalKMS = kms.NewStub("default-test-key")
|
||
|
}
|
||
|
|
||
|
tearDown := func() {
|
||
|
adminTestBed.TearDown()
|
||
|
GlobalKMS = nil
|
||
|
}
|
||
|
return adminTestBed, tearDown
|
||
|
}
|
||
|
|
||
|
func buildKMSRequest(t *testing.T, method, path, accessKey, secretKey string, query map[string]string) *http.Request {
|
||
|
if len(query) > 0 {
|
||
|
queryVal := url.Values{}
|
||
|
for k, v := range query {
|
||
|
queryVal.Add(k, v)
|
||
|
}
|
||
|
path = path + "?" + queryVal.Encode()
|
||
|
}
|
||
|
|
||
|
if accessKey == "" && secretKey == "" {
|
||
|
accessKey = globalActiveCred.AccessKey
|
||
|
secretKey = globalActiveCred.SecretKey
|
||
|
}
|
||
|
|
||
|
req, err := newTestSignedRequestV4(method, path, 0, nil, accessKey, secretKey, nil)
|
||
|
if err != nil {
|
||
|
t.Fatal(err)
|
||
|
}
|
||
|
return req
|
||
|
}
|
||
|
|
||
|
// setupKMSUser is a test helper that creates a new user with the provided access key and secret key
|
||
|
// and applies the given policy to the user.
|
||
|
func setupKMSUser(t *testing.T, accessKey, secretKey, p string) {
|
||
|
ctx := context.Background()
|
||
|
createUserParams := madmin.AddOrUpdateUserReq{
|
||
|
SecretKey: secretKey,
|
||
|
Status: madmin.AccountEnabled,
|
||
|
}
|
||
|
_, err := globalIAMSys.CreateUser(ctx, accessKey, createUserParams)
|
||
|
if err != nil {
|
||
|
t.Fatal(err)
|
||
|
}
|
||
|
|
||
|
testKMSPolicyName := "testKMSPolicy"
|
||
|
if p != "" {
|
||
|
p = `{"Version":"2012-10-17","Statement":[` + p + `]}`
|
||
|
policyData, err := policy.ParseConfig(strings.NewReader(p))
|
||
|
if err != nil {
|
||
|
t.Fatal(err)
|
||
|
}
|
||
|
_, err = globalIAMSys.SetPolicy(ctx, testKMSPolicyName, *policyData)
|
||
|
if err != nil {
|
||
|
t.Fatal(err)
|
||
|
}
|
||
|
_, err = globalIAMSys.PolicyDBSet(ctx, accessKey, testKMSPolicyName, regUser, false)
|
||
|
if err != nil {
|
||
|
t.Fatal(err)
|
||
|
}
|
||
|
} else {
|
||
|
err = globalIAMSys.DeletePolicy(ctx, testKMSPolicyName, false)
|
||
|
if err != nil {
|
||
|
t.Fatal(err)
|
||
|
}
|
||
|
_, err = globalIAMSys.PolicyDBSet(ctx, accessKey, "", regUser, false)
|
||
|
if err != nil {
|
||
|
t.Fatal(err)
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func keyNamesFromListKeysResp(t *testing.T, b []byte) []string {
|
||
|
var keyInfos []madmin.KMSKeyInfo
|
||
|
err := json.Unmarshal(b, &keyInfos)
|
||
|
if err != nil {
|
||
|
t.Fatalf("cannot unmarshal '%s', err: %v", b, err)
|
||
|
}
|
||
|
|
||
|
var gotKeyNames []string
|
||
|
for _, keyInfo := range keyInfos {
|
||
|
gotKeyNames = append(gotKeyNames, keyInfo.Name)
|
||
|
}
|
||
|
return gotKeyNames
|
||
|
}
|