2014-12-17 19:35:35 +00:00
|
|
|
// Package auth defines a standard interface for request access controllers.
|
|
|
|
//
|
|
|
|
// An access controller has a simple interface with a single `Authorized`
|
|
|
|
// method which checks that a given request is authorized to perform one or
|
|
|
|
// more actions on one or more resources. This method should return a non-nil
|
2015-04-17 12:39:52 +00:00
|
|
|
// error if the request is not authorized.
|
2014-12-17 19:35:35 +00:00
|
|
|
//
|
|
|
|
// An implementation registers its access controller by name with a constructor
|
|
|
|
// which accepts an options map for configuring the access controller.
|
|
|
|
//
|
|
|
|
// options := map[string]interface{}{"sillySecret": "whysosilly?"}
|
|
|
|
// accessController, _ := auth.GetAccessController("silly", options)
|
|
|
|
//
|
|
|
|
// This `accessController` can then be used in a request handler like so:
|
|
|
|
//
|
|
|
|
// func updateOrder(w http.ResponseWriter, r *http.Request) {
|
|
|
|
// orderNumber := r.FormValue("orderNumber")
|
|
|
|
// resource := auth.Resource{Type: "customerOrder", Name: orderNumber}
|
|
|
|
// access := auth.Access{Resource: resource, Action: "update"}
|
|
|
|
//
|
2015-02-04 01:59:24 +00:00
|
|
|
// if ctx, err := accessController.Authorized(ctx, access); err != nil {
|
2014-12-17 19:35:35 +00:00
|
|
|
// if challenge, ok := err.(auth.Challenge) {
|
|
|
|
// // Let the challenge write the response.
|
2015-10-20 13:57:15 +00:00
|
|
|
// challenge.SetHeaders(w)
|
|
|
|
// w.WriteHeader(http.StatusUnauthorized)
|
|
|
|
// return
|
2014-12-17 19:35:35 +00:00
|
|
|
// } else {
|
|
|
|
// // Some other error.
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
//
|
2014-12-17 06:58:39 +00:00
|
|
|
package auth
|
|
|
|
|
|
|
|
import (
|
2017-08-11 22:31:16 +00:00
|
|
|
"context"
|
2016-02-13 01:15:19 +00:00
|
|
|
"errors"
|
2014-12-17 06:58:39 +00:00
|
|
|
"fmt"
|
|
|
|
"net/http"
|
2015-02-04 01:59:24 +00:00
|
|
|
)
|
|
|
|
|
2016-01-29 01:02:09 +00:00
|
|
|
const (
|
|
|
|
// UserKey is used to get the user object from
|
|
|
|
// a user context
|
|
|
|
UserKey = "auth.user"
|
|
|
|
|
|
|
|
// UserNameKey is used to get the user name from
|
|
|
|
// a user context
|
|
|
|
UserNameKey = "auth.user.name"
|
|
|
|
)
|
|
|
|
|
2016-02-13 01:15:19 +00:00
|
|
|
var (
|
|
|
|
// ErrInvalidCredential is returned when the auth token does not authenticate correctly.
|
|
|
|
ErrInvalidCredential = errors.New("invalid authorization credential")
|
|
|
|
|
2016-03-04 21:53:06 +00:00
|
|
|
// ErrAuthenticationFailure returned when authentication fails.
|
2016-02-13 01:15:19 +00:00
|
|
|
ErrAuthenticationFailure = errors.New("authentication failure")
|
|
|
|
)
|
|
|
|
|
2015-02-04 01:59:24 +00:00
|
|
|
// UserInfo carries information about
|
|
|
|
// an autenticated/authorized client.
|
|
|
|
type UserInfo struct {
|
|
|
|
Name string
|
|
|
|
}
|
|
|
|
|
2014-12-17 06:58:39 +00:00
|
|
|
// Resource describes a resource by type and name.
|
|
|
|
type Resource struct {
|
2016-11-16 01:14:31 +00:00
|
|
|
Type string
|
|
|
|
Class string
|
|
|
|
Name string
|
2014-12-17 06:58:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Access describes a specific action that is
|
2015-04-17 12:39:52 +00:00
|
|
|
// requested or allowed for a given resource.
|
2014-12-17 06:58:39 +00:00
|
|
|
type Access struct {
|
|
|
|
Resource
|
|
|
|
Action string
|
|
|
|
}
|
|
|
|
|
|
|
|
// Challenge is a special error type which is used for HTTP 401 Unauthorized
|
|
|
|
// responses and is able to write the response with WWW-Authenticate challenge
|
|
|
|
// header values based on the error.
|
|
|
|
type Challenge interface {
|
|
|
|
error
|
2015-07-24 02:39:56 +00:00
|
|
|
|
|
|
|
// SetHeaders prepares the request to conduct a challenge response by
|
|
|
|
// adding the an HTTP challenge header on the response message. Callers
|
|
|
|
// are expected to set the appropriate HTTP status code (e.g. 401)
|
|
|
|
// themselves.
|
|
|
|
SetHeaders(w http.ResponseWriter)
|
2014-12-17 06:58:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// AccessController controls access to registry resources based on a request
|
|
|
|
// and required access levels for a request. Implementations can support both
|
|
|
|
// complete denial and http authorization challenges.
|
|
|
|
type AccessController interface {
|
2015-02-04 01:59:24 +00:00
|
|
|
// Authorized returns a non-nil error if the context is granted access and
|
|
|
|
// returns a new authorized context. If one or more Access structs are
|
|
|
|
// provided, the requested access will be compared with what is available
|
|
|
|
// to the context. The given context will contain a "http.request" key with
|
|
|
|
// a `*http.Request` value. If the error is non-nil, access should always
|
|
|
|
// be denied. The error may be of type Challenge, in which case the caller
|
|
|
|
// may have the Challenge handle the request or choose what action to take
|
|
|
|
// based on the Challenge header or response status. The returned context
|
|
|
|
// object should have a "auth.user" value set to a UserInfo struct.
|
|
|
|
Authorized(ctx context.Context, access ...Access) (context.Context, error)
|
2014-12-17 06:58:39 +00:00
|
|
|
}
|
|
|
|
|
2016-03-04 21:53:06 +00:00
|
|
|
// CredentialAuthenticator is an object which is able to authenticate credentials
|
2016-02-13 01:15:19 +00:00
|
|
|
type CredentialAuthenticator interface {
|
|
|
|
AuthenticateUser(username, password string) error
|
|
|
|
}
|
|
|
|
|
2015-02-06 03:12:32 +00:00
|
|
|
// WithUser returns a context with the authorized user info.
|
|
|
|
func WithUser(ctx context.Context, user UserInfo) context.Context {
|
|
|
|
return userInfoContext{
|
|
|
|
Context: ctx,
|
|
|
|
user: user,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
type userInfoContext struct {
|
|
|
|
context.Context
|
|
|
|
user UserInfo
|
|
|
|
}
|
|
|
|
|
|
|
|
func (uic userInfoContext) Value(key interface{}) interface{} {
|
|
|
|
switch key {
|
2016-01-29 01:02:09 +00:00
|
|
|
case UserKey:
|
2015-02-06 03:12:32 +00:00
|
|
|
return uic.user
|
2016-01-29 01:02:09 +00:00
|
|
|
case UserNameKey:
|
2015-02-06 03:12:32 +00:00
|
|
|
return uic.user.Name
|
|
|
|
}
|
|
|
|
|
|
|
|
return uic.Context.Value(key)
|
|
|
|
}
|
|
|
|
|
2016-11-22 00:36:36 +00:00
|
|
|
// WithResources returns a context with the authorized resources.
|
|
|
|
func WithResources(ctx context.Context, resources []Resource) context.Context {
|
|
|
|
return resourceContext{
|
|
|
|
Context: ctx,
|
|
|
|
resources: resources,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
type resourceContext struct {
|
|
|
|
context.Context
|
|
|
|
resources []Resource
|
|
|
|
}
|
|
|
|
|
|
|
|
type resourceKey struct{}
|
|
|
|
|
|
|
|
func (rc resourceContext) Value(key interface{}) interface{} {
|
|
|
|
if key == (resourceKey{}) {
|
|
|
|
return rc.resources
|
|
|
|
}
|
|
|
|
|
|
|
|
return rc.Context.Value(key)
|
|
|
|
}
|
|
|
|
|
|
|
|
// AuthorizedResources returns the list of resources which have
|
|
|
|
// been authorized for this request.
|
|
|
|
func AuthorizedResources(ctx context.Context) []Resource {
|
|
|
|
if resources, ok := ctx.Value(resourceKey{}).([]Resource); ok {
|
|
|
|
return resources
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2014-12-17 06:58:39 +00:00
|
|
|
// InitFunc is the type of an AccessController factory function and is used
|
2015-03-09 16:42:23 +00:00
|
|
|
// to register the constructor for different AccesController backends.
|
2014-12-17 06:58:39 +00:00
|
|
|
type InitFunc func(options map[string]interface{}) (AccessController, error)
|
|
|
|
|
|
|
|
var accessControllers map[string]InitFunc
|
|
|
|
|
|
|
|
func init() {
|
|
|
|
accessControllers = make(map[string]InitFunc)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Register is used to register an InitFunc for
|
|
|
|
// an AccessController backend with the given name.
|
|
|
|
func Register(name string, initFunc InitFunc) error {
|
|
|
|
if _, exists := accessControllers[name]; exists {
|
|
|
|
return fmt.Errorf("name already registered: %s", name)
|
|
|
|
}
|
|
|
|
|
|
|
|
accessControllers[name] = initFunc
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetAccessController constructs an AccessController
|
|
|
|
// with the given options using the named backend.
|
|
|
|
func GetAccessController(name string, options map[string]interface{}) (AccessController, error) {
|
|
|
|
if initFunc, exists := accessControllers[name]; exists {
|
|
|
|
return initFunc(options)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil, fmt.Errorf("no access controller registered with name: %s", name)
|
|
|
|
}
|