2017-02-14 21:14:24 +00:00
|
|
|
package oauth2
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"errors"
|
|
|
|
"net/http"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"golang.org/x/oauth2"
|
|
|
|
)
|
|
|
|
|
2017-02-24 05:26:09 +00:00
|
|
|
type principalKey string
|
|
|
|
|
|
|
|
func (p principalKey) String() string {
|
|
|
|
return string(p)
|
|
|
|
}
|
|
|
|
|
|
|
|
var (
|
2017-02-14 21:14:24 +00:00
|
|
|
// PrincipalKey is used to pass principal
|
|
|
|
// via context.Context to request-scoped
|
|
|
|
// functions.
|
2017-02-24 05:26:09 +00:00
|
|
|
PrincipalKey = principalKey("principal")
|
|
|
|
// ErrAuthentication means that oauth2 exchange failed
|
2017-02-14 21:14:24 +00:00
|
|
|
ErrAuthentication = errors.New("user not authenticated")
|
2017-02-24 05:26:09 +00:00
|
|
|
// ErrOrgMembership means that the user is not in the OAuth2 filtered group
|
|
|
|
ErrOrgMembership = errors.New("Not a member of the required organization")
|
2017-02-14 21:14:24 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
/* Types */
|
|
|
|
|
|
|
|
// Principal is any entity that can be authenticated
|
2017-02-15 22:28:17 +00:00
|
|
|
type Principal struct {
|
2017-10-26 22:01:20 +00:00
|
|
|
Subject string
|
|
|
|
Issuer string
|
|
|
|
Organization string
|
2018-01-02 23:48:58 +00:00
|
|
|
Group string
|
2017-10-26 22:01:20 +00:00
|
|
|
ExpiresAt time.Time
|
|
|
|
IssuedAt time.Time
|
2017-02-15 22:28:17 +00:00
|
|
|
}
|
2017-02-14 21:14:24 +00:00
|
|
|
|
|
|
|
/* Interfaces */
|
|
|
|
|
|
|
|
// Provider are the common parameters for all providers (RFC 6749)
|
|
|
|
type Provider interface {
|
|
|
|
// ID is issued to the registered client by the authorization (RFC 6749 Section 2.2)
|
|
|
|
ID() string
|
|
|
|
// Secret associated is with the ID (Section 2.2)
|
|
|
|
Secret() string
|
|
|
|
// Scopes is used by the authorization server to "scope" responses (Section 3.3)
|
|
|
|
Scopes() []string
|
|
|
|
// Config is the OAuth2 configuration settings for this provider
|
|
|
|
Config() *oauth2.Config
|
|
|
|
// PrincipalID with fetch the identifier to be associated with the principal.
|
|
|
|
PrincipalID(provider *http.Client) (string, error)
|
2017-02-14 22:28:05 +00:00
|
|
|
// Name is the name of the Provider
|
|
|
|
Name() string
|
2018-01-02 23:48:58 +00:00
|
|
|
// Group is a comma delimited list of groups and organizations for a provider
|
|
|
|
// TODO: This will break if there are any group names that contain commas.
|
|
|
|
// I think this is okay, but I'm not 100% certain.
|
|
|
|
Group(provider *http.Client) (string, error)
|
2017-02-14 21:14:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Mux is a collection of handlers responsible for servicing an Oauth2 interaction between a browser and a provider
|
|
|
|
type Mux interface {
|
|
|
|
Login() http.Handler
|
|
|
|
Logout() http.Handler
|
|
|
|
Callback() http.Handler
|
|
|
|
}
|
|
|
|
|
|
|
|
// Authenticator represents a service for authenticating users.
|
|
|
|
type Authenticator interface {
|
2017-04-06 18:40:57 +00:00
|
|
|
// Validate returns Principal associated with authenticated and authorized
|
2017-04-17 16:49:45 +00:00
|
|
|
// entity if successful.
|
|
|
|
Validate(context.Context, *http.Request) (Principal, error)
|
2017-04-06 18:40:57 +00:00
|
|
|
// Authorize will grant privileges to a Principal
|
|
|
|
Authorize(context.Context, http.ResponseWriter, Principal) error
|
2017-04-17 16:49:45 +00:00
|
|
|
// Extend will extend the lifetime of a already validated Principal
|
|
|
|
Extend(context.Context, http.ResponseWriter, Principal) (Principal, error)
|
2017-04-06 18:40:57 +00:00
|
|
|
// Expire revokes privileges from a Principal
|
|
|
|
Expire(http.ResponseWriter)
|
2017-02-14 21:14:24 +00:00
|
|
|
}
|
|
|
|
|
2017-04-06 18:40:57 +00:00
|
|
|
// Token represents a time-dependent reference (i.e. identifier) that maps back
|
|
|
|
// to the sensitive data through a tokenization system
|
|
|
|
type Token string
|
|
|
|
|
|
|
|
// Tokenizer substitutes a sensitive data element (Principal) with a
|
|
|
|
// non-sensitive equivalent, referred to as a token, that has no extrinsic
|
|
|
|
// or exploitable meaning or value.
|
|
|
|
type Tokenizer interface {
|
2017-04-14 07:12:52 +00:00
|
|
|
// Create issues a token at Principal's IssuedAt that lasts until Principal's ExpireAt
|
|
|
|
Create(context.Context, Principal) (Token, error)
|
2017-04-06 18:40:57 +00:00
|
|
|
// ValidPrincipal checks if the token has a valid Principal and requires
|
2017-04-14 07:12:52 +00:00
|
|
|
// a lifespan duration to ensure it complies with possible server runtime arguments.
|
|
|
|
ValidPrincipal(ctx context.Context, token Token, lifespan time.Duration) (Principal, error)
|
2017-04-17 16:49:45 +00:00
|
|
|
// ExtendedPrincipal adds the extention to the principal's lifespan.
|
|
|
|
ExtendedPrincipal(ctx context.Context, principal Principal, extension time.Duration) (Principal, error)
|
2017-02-14 21:14:24 +00:00
|
|
|
}
|