2018-09-05 19:06:26 +00:00
|
|
|
package query
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2018-09-06 18:09:52 +00:00
|
|
|
|
|
|
|
"github.com/influxdata/flux"
|
|
|
|
"github.com/influxdata/platform"
|
|
|
|
"github.com/pkg/errors"
|
2018-09-05 19:06:26 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// PreAuthorizer provides a method for ensuring that the buckets accessed by a query spec
|
|
|
|
// are allowed access by the given Authorization. This is a pre-check provided as a way for
|
|
|
|
// callers to fail early for operations that are not allowed. However, it's still possible
|
|
|
|
// for authorization to be denied at runtime even if this check passes.
|
|
|
|
type PreAuthorizer interface {
|
2018-09-06 18:09:52 +00:00
|
|
|
PreAuthorize(ctx context.Context, spec *flux.Spec, auth *platform.Authorization) error
|
2018-09-05 19:06:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// NewPreAuthorizer creates a new PreAuthorizer
|
|
|
|
func NewPreAuthorizer(bucketService platform.BucketService) PreAuthorizer {
|
|
|
|
return &preAuthorizer{bucketService: bucketService}
|
|
|
|
}
|
|
|
|
|
|
|
|
type preAuthorizer struct {
|
|
|
|
bucketService platform.BucketService
|
|
|
|
}
|
|
|
|
|
|
|
|
// PreAuthorize finds all the buckets read and written by the given spec, and ensures that execution is allowed
|
|
|
|
// given the Authorization. Returns nil on success, and an error with an appropriate message otherwise.
|
2018-09-06 18:09:52 +00:00
|
|
|
func (a *preAuthorizer) PreAuthorize(ctx context.Context, spec *flux.Spec, auth *platform.Authorization) error {
|
2018-09-05 19:06:26 +00:00
|
|
|
|
2018-09-11 22:56:51 +00:00
|
|
|
readBuckets, writeBuckets, err := BucketsAccessed(spec)
|
2018-09-05 19:06:26 +00:00
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "Could not retrieve buckets for query.Spec")
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, readBucketFilter := range readBuckets {
|
|
|
|
bucket, err := a.bucketService.FindBucket(ctx, readBucketFilter)
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrapf(err, "Bucket service error")
|
|
|
|
} else if bucket == nil {
|
2018-11-21 14:22:35 +00:00
|
|
|
return errors.New("bucket service returned nil bucket")
|
2018-09-05 19:06:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
reqPerm := platform.ReadBucketPermission(bucket.ID)
|
2018-09-28 17:02:34 +00:00
|
|
|
if !auth.Allowed(reqPerm) {
|
2018-11-21 14:22:35 +00:00
|
|
|
return errors.New("no read permission for bucket: \"" + bucket.Name + "\"")
|
2018-09-05 19:06:26 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, writeBucketFilter := range writeBuckets {
|
|
|
|
bucket, err := a.bucketService.FindBucket(context.Background(), writeBucketFilter)
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrapf(err, "Could not find bucket %v", writeBucketFilter)
|
|
|
|
}
|
|
|
|
|
|
|
|
reqPerm := platform.WriteBucketPermission(bucket.ID)
|
2018-09-28 17:02:34 +00:00
|
|
|
if !auth.Allowed(reqPerm) {
|
2018-11-21 14:22:35 +00:00
|
|
|
return errors.New("no write permission for bucket: \"" + bucket.Name + "\"")
|
2018-09-05 19:06:26 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|