mirror of
https://github.com/minio/minio.git
synced 2025-01-13 16:03:21 -05:00
Add policy validation code more rigorous
This commit is contained in:
parent
7882c8c746
commit
980d229272
@ -144,3 +144,17 @@ func (c *Config) ReadConfig() error {
|
||||
c.Users = users
|
||||
return nil
|
||||
}
|
||||
|
||||
func Loadusers() map[string]User {
|
||||
c := Config{}
|
||||
c.SetupConfig()
|
||||
c.ReadConfig()
|
||||
return c.Users
|
||||
}
|
||||
|
||||
func Loaduser(username string) User {
|
||||
c := Config{}
|
||||
c.SetupConfig()
|
||||
c.ReadConfig()
|
||||
return c.GetUser(username)
|
||||
}
|
||||
|
@ -24,7 +24,7 @@ func (d Date) IsZero() bool {
|
||||
|
||||
// Convert string date in format YYYY-MM-DD to Date.
|
||||
// Leading and trailing spaces are ignored. If format is invalid returns zero.
|
||||
func ParseDate(str string) (d Date, err error) {
|
||||
func parseDate(str string) (d Date, err error) {
|
||||
str = strings.TrimSpace(str)
|
||||
if str == "0000-00-00" {
|
||||
return
|
||||
|
@ -3,6 +3,7 @@ package policy
|
||||
import (
|
||||
"encoding/json"
|
||||
"io"
|
||||
"strings"
|
||||
)
|
||||
|
||||
type UserCred struct {
|
||||
@ -15,6 +16,7 @@ type Stmt struct {
|
||||
Principal UserCred
|
||||
Action []string
|
||||
Resource []string
|
||||
// TODO fix it in future if necessary - Condition {}
|
||||
}
|
||||
|
||||
type BucketPolicy struct {
|
||||
@ -22,7 +24,97 @@ type BucketPolicy struct {
|
||||
Statement []Stmt
|
||||
}
|
||||
|
||||
// TODO: Add more checks
|
||||
const (
|
||||
awsResource = "arn:aws:s3:::"
|
||||
minioResource = "minio:::"
|
||||
)
|
||||
|
||||
// TODO support canonical user
|
||||
const (
|
||||
awsPrincipal = "arn:aws:iam::Account-ID:user/"
|
||||
minioPrincipal = "minio::Account-ID:user/"
|
||||
)
|
||||
|
||||
var supportedActionMap = map[string]bool{
|
||||
"*": true,
|
||||
"s3:GetObject": true,
|
||||
"s3:ListBucket": true,
|
||||
"s3:PutObject": true,
|
||||
"s3:CreateBucket": true,
|
||||
"s3:GetBucketPolicy": true,
|
||||
"s3:DeleteBucketPolicy": true,
|
||||
"s3:ListAllMyBuckets": true,
|
||||
"s3:PutBucketPolicy": true,
|
||||
}
|
||||
|
||||
var supportedEffectMap = map[string]bool{
|
||||
"Allow": true,
|
||||
"Deny": true,
|
||||
}
|
||||
|
||||
func isValidAction(action []string) bool {
|
||||
var ok bool = false
|
||||
for _, a := range action {
|
||||
if supportedActionMap[a] {
|
||||
ok = true
|
||||
}
|
||||
}
|
||||
return ok
|
||||
}
|
||||
|
||||
func isValidEffect(effect string) bool {
|
||||
if supportedEffectMap[effect] {
|
||||
return true
|
||||
}
|
||||
return false
|
||||
}
|
||||
|
||||
func isValidResource(resources []string) bool {
|
||||
var ok bool = false
|
||||
for _, resource := range resources {
|
||||
switch true {
|
||||
case strings.HasPrefix(resource, awsResource):
|
||||
bucket := strings.SplitAfter(resource, awsResource)[1]
|
||||
ok = true
|
||||
if len(bucket) == 0 {
|
||||
ok = false
|
||||
}
|
||||
case strings.HasPrefix(resource, minioResource):
|
||||
bucket := strings.SplitAfter(resource, minioResource)[1]
|
||||
ok = true
|
||||
if len(bucket) == 0 {
|
||||
ok = false
|
||||
}
|
||||
default:
|
||||
ok = false
|
||||
}
|
||||
}
|
||||
return ok
|
||||
}
|
||||
|
||||
func isValidPrincipal(principal string) bool {
|
||||
var ok bool = false
|
||||
if principal == "*" {
|
||||
return true
|
||||
}
|
||||
switch true {
|
||||
case strings.HasPrefix(principal, awsPrincipal):
|
||||
username := strings.SplitAfter(principal, awsPrincipal)[1]
|
||||
ok = true
|
||||
if len(username) == 0 {
|
||||
ok = false
|
||||
}
|
||||
case strings.HasPrefix(principal, minioPrincipal):
|
||||
username := strings.SplitAfter(principal, minioPrincipal)[1]
|
||||
ok = true
|
||||
if len(username) == 0 {
|
||||
ok = false
|
||||
}
|
||||
default:
|
||||
ok = false
|
||||
}
|
||||
return ok
|
||||
}
|
||||
|
||||
// validate request body is proper JSON
|
||||
func Parsepolicy(data io.Reader) (BucketPolicy, bool) {
|
||||
@ -35,7 +127,7 @@ func Parsepolicy(data io.Reader) (BucketPolicy, bool) {
|
||||
if len(policy.Version) == 0 {
|
||||
goto error
|
||||
}
|
||||
_, err = ParseDate(policy.Version)
|
||||
_, err = parseDate(policy.Version)
|
||||
if err != nil {
|
||||
goto error
|
||||
}
|
||||
@ -50,15 +142,27 @@ func Parsepolicy(data io.Reader) (BucketPolicy, bool) {
|
||||
if len(statement.Effect) == 0 {
|
||||
goto error
|
||||
}
|
||||
if !isValidEffect(statement.Effect) {
|
||||
goto error
|
||||
}
|
||||
if len(statement.Principal.AWS) == 0 {
|
||||
goto error
|
||||
}
|
||||
if !isValidPrincipal(statement.Principal.AWS) {
|
||||
goto error
|
||||
}
|
||||
if len(statement.Action) == 0 {
|
||||
goto error
|
||||
}
|
||||
if !isValidAction(statement.Action) {
|
||||
goto error
|
||||
}
|
||||
if len(statement.Resource) == 0 {
|
||||
goto error
|
||||
}
|
||||
if !isValidResource(statement.Resource) {
|
||||
goto error
|
||||
}
|
||||
}
|
||||
return policy, true
|
||||
|
||||
|
Loading…
Reference in New Issue
Block a user