mirror of
https://github.com/seaweedfs/seaweedfs.git
synced 2026-06-13 23:36:45 +03:00
8e4022d5c7
Bulk DeleteObjects carries the keys in the request body, so the route Auth middleware ran a single bucket-level check with object="", building the resource ARN as arn:aws:s3:::<bucket>. That never matches an s3:DeleteObject policy scoped to <bucket>/*, so the entire batch was denied even though the single-key DELETE worked with the same credentials. Defer authorization to the handler and check each key via AuthorizeBatchDeleteKey, mirroring AuthorizeCopySource: a synthetic DELETE /<bucket>/<key> request resolves s3:DeleteObject (or s3:DeleteObjectVersion when a versionId is given) against the object ARN. Denied keys come back as per-key errors while authorized keys still delete, matching AWS semantics.
94 lines
2.8 KiB
Go
94 lines
2.8 KiB
Go
package s3api
|
|
|
|
import (
|
|
"encoding/json"
|
|
"net/http/httptest"
|
|
"testing"
|
|
|
|
"github.com/seaweedfs/seaweedfs/weed/s3api/s3err"
|
|
"github.com/stretchr/testify/require"
|
|
)
|
|
|
|
// TestAuthorizeBatchDeleteKey_AwsCanonicalPolicy: a policy granting s3:DeleteObject
|
|
// on <bucket>/* must allow per-key batch deletes. Pre-fix the bucket-level check
|
|
// built arn:aws:s3:::<bucket> and never matched the object-scoped policy.
|
|
func TestAuthorizeBatchDeleteKey_AwsCanonicalPolicy(t *testing.T) {
|
|
const bucket = "test-bucket"
|
|
const policyName = "delete-test-bucket-objects"
|
|
|
|
policyDoc, err := json.Marshal(map[string]any{
|
|
"Version": "2012-10-17",
|
|
"Statement": []map[string]any{
|
|
{
|
|
"Effect": "Allow",
|
|
"Action": "s3:DeleteObject",
|
|
"Resource": "arn:aws:s3:::" + bucket + "/*",
|
|
},
|
|
},
|
|
})
|
|
require.NoError(t, err)
|
|
|
|
iam := &IdentityAccessManagement{
|
|
isAuthEnabled: true,
|
|
}
|
|
require.NoError(t, iam.PutPolicy(policyName, string(policyDoc)))
|
|
|
|
identity := &Identity{
|
|
Name: "alice",
|
|
Account: &AccountAdmin,
|
|
PolicyNames: []string{policyName},
|
|
Credentials: []*Credential{{AccessKey: "AKIAEXAMPLE", SecretKey: "secret"}},
|
|
}
|
|
|
|
r := httptest.NewRequest("POST", "/"+bucket+"?delete", nil)
|
|
|
|
require.Equal(t, s3err.ErrNone,
|
|
iam.AuthorizeBatchDeleteKey(r, identity, bucket, "objects/a.txt", ""),
|
|
"s3:DeleteObject on arn:aws:s3:::%s/* must allow deleting %s/objects/a.txt", bucket, bucket)
|
|
|
|
require.Equal(t, s3err.ErrAccessDenied,
|
|
iam.AuthorizeBatchDeleteKey(r, identity, "other-bucket", "objects/a.txt", ""),
|
|
"keys outside the granted bucket must be denied")
|
|
}
|
|
|
|
// TestAuthorizeBatchDeleteKey_PrefixScopedPolicy: a prefix-scoped policy must allow
|
|
// batch deletes under the prefix and deny keys outside it, per-key.
|
|
func TestAuthorizeBatchDeleteKey_PrefixScopedPolicy(t *testing.T) {
|
|
const bucket = "test-bucket"
|
|
const policyName = "delete-prefix-only"
|
|
|
|
policyDoc, err := json.Marshal(map[string]any{
|
|
"Version": "2012-10-17",
|
|
"Statement": []map[string]any{
|
|
{
|
|
"Effect": "Allow",
|
|
"Action": "s3:DeleteObject",
|
|
"Resource": "arn:aws:s3:::" + bucket + "/safe/*",
|
|
},
|
|
},
|
|
})
|
|
require.NoError(t, err)
|
|
|
|
iam := &IdentityAccessManagement{
|
|
isAuthEnabled: true,
|
|
}
|
|
require.NoError(t, iam.PutPolicy(policyName, string(policyDoc)))
|
|
|
|
identity := &Identity{
|
|
Name: "alice",
|
|
Account: &AccountAdmin,
|
|
PolicyNames: []string{policyName},
|
|
Credentials: []*Credential{{AccessKey: "AKIAEXAMPLE", SecretKey: "secret"}},
|
|
}
|
|
|
|
r := httptest.NewRequest("POST", "/"+bucket+"?delete", nil)
|
|
|
|
require.Equal(t, s3err.ErrNone,
|
|
iam.AuthorizeBatchDeleteKey(r, identity, bucket, "safe/inside.txt", ""),
|
|
"key under granted prefix must be allowed")
|
|
|
|
require.Equal(t, s3err.ErrAccessDenied,
|
|
iam.AuthorizeBatchDeleteKey(r, identity, bucket, "danger/outside.txt", ""),
|
|
"key outside the granted prefix must be denied per-key, not at the batch level")
|
|
}
|