2015-01-03 22:03:57 +02:00
|
|
|
package authboss
|
|
|
|
|
|
|
|
import (
|
2015-01-19 00:35:44 +02:00
|
|
|
"bytes"
|
2017-02-21 00:28:38 +02:00
|
|
|
"context"
|
2015-01-03 22:03:57 +02:00
|
|
|
"time"
|
2017-02-22 01:04:30 +02:00
|
|
|
|
|
|
|
"github.com/pkg/errors"
|
2015-01-03 22:03:57 +02:00
|
|
|
)
|
|
|
|
|
2015-02-10 10:43:45 +02:00
|
|
|
// Data store constants for attribute names.
|
|
|
|
const (
|
2015-02-16 23:27:29 +02:00
|
|
|
StoreEmail = "email"
|
|
|
|
StoreUsername = "username"
|
|
|
|
StorePassword = "password"
|
2015-02-10 10:43:45 +02:00
|
|
|
)
|
|
|
|
|
2015-03-13 04:20:36 +02:00
|
|
|
// Data store constants for OAuth2 attribute names.
|
|
|
|
const (
|
2015-03-14 01:23:43 +02:00
|
|
|
StoreOAuth2UID = "oauth2_uid"
|
|
|
|
StoreOAuth2Provider = "oauth2_provider"
|
|
|
|
StoreOAuth2Token = "oauth2_token"
|
|
|
|
StoreOAuth2Refresh = "oauth2_refresh"
|
|
|
|
StoreOAuth2Expiry = "oauth2_expiry"
|
2015-03-13 04:20:36 +02:00
|
|
|
)
|
|
|
|
|
2015-01-24 01:56:24 +02:00
|
|
|
var (
|
2015-02-02 00:17:18 +02:00
|
|
|
// ErrUserNotFound should be returned from Get when the record is not found.
|
2017-02-22 01:04:30 +02:00
|
|
|
ErrUserNotFound = errors.New("user not found")
|
2015-02-02 00:17:18 +02:00
|
|
|
// ErrTokenNotFound should be returned from UseToken when the record is not found.
|
2017-02-22 01:04:30 +02:00
|
|
|
ErrTokenNotFound = errors.New("token not found")
|
2017-02-21 00:28:38 +02:00
|
|
|
// ErrUserFound should be returned from Create when the primaryID of the record is found.
|
2017-02-22 01:04:30 +02:00
|
|
|
ErrUserFound = errors.New("user found")
|
2015-01-24 01:56:24 +02:00
|
|
|
)
|
2015-01-11 08:52:39 +02:00
|
|
|
|
2017-02-21 00:28:38 +02:00
|
|
|
// StoreLoader represents the data store that's capable of loading users
|
|
|
|
// and giving them a context with which to store themselves.
|
|
|
|
type StoreLoader interface {
|
|
|
|
// Load will be passed the PrimaryID and return pre-loaded storer (meaning
|
|
|
|
// Storer.Load will not be called)
|
|
|
|
Load(ctx context.Context, key string) (Storer, error)
|
|
|
|
}
|
2015-01-11 08:52:39 +02:00
|
|
|
|
2017-02-21 00:28:38 +02:00
|
|
|
// Storer represents a user that also knows how to put himself into the db.
|
|
|
|
// It has functions for each piece of data it requires.
|
|
|
|
// Note that you should only persist data once Save() has been called.
|
2015-01-11 08:52:39 +02:00
|
|
|
type Storer interface {
|
2017-02-21 00:28:38 +02:00
|
|
|
PutEmail(ctx context.Context, email string) error
|
|
|
|
PutUsername(ctx context.Context, username string) error
|
|
|
|
PutPassword(ctx context.Context, password string) error
|
|
|
|
|
|
|
|
GetEmail(ctx context.Context) (email string, err error)
|
|
|
|
GetUsername(ctx context.Context) (username string, err error)
|
|
|
|
GetPassword(ctx context.Context) (password string, err error)
|
|
|
|
|
2017-02-21 02:08:19 +02:00
|
|
|
// Save the state.
|
2017-02-21 00:28:38 +02:00
|
|
|
Save(ctx context.Context) error
|
|
|
|
|
|
|
|
// Load the state based on the properties that have been given (typically
|
|
|
|
// an e-mail/username).
|
|
|
|
Load(ctx context.Context) error
|
|
|
|
}
|
|
|
|
|
2017-02-25 02:45:47 +02:00
|
|
|
// TODO(aarondl): Document & move to Register module
|
|
|
|
// ArbitraryStorer allows arbitrary data from the web form through. You should
|
|
|
|
// definitely only pull the keys you want from the map, since this is unfiltered
|
|
|
|
// input from a web request and is an attack vector.
|
2017-02-21 00:28:38 +02:00
|
|
|
type ArbitraryStorer interface {
|
|
|
|
Storer
|
|
|
|
|
|
|
|
// PutArbitrary allows arbitrary fields defined by the authboss library
|
|
|
|
// consumer to add fields to the user registration piece.
|
2017-02-21 01:56:26 +02:00
|
|
|
PutArbitrary(ctx context.Context, arbitrary map[string]string) error
|
2017-02-21 00:28:38 +02:00
|
|
|
// GetArbitrary is used only to display the arbitrary data back to the user
|
|
|
|
// when the form is reset.
|
|
|
|
GetArbitrary(ctx context.Context) (arbitrary map[string]string, err error)
|
2015-01-11 08:52:39 +02:00
|
|
|
}
|
|
|
|
|
2017-02-24 02:13:25 +02:00
|
|
|
// OAuth2Storer allows reading and writing values relating to OAuth2
|
2015-03-14 01:23:43 +02:00
|
|
|
type OAuth2Storer interface {
|
2017-02-21 00:28:38 +02:00
|
|
|
Storer
|
|
|
|
|
2017-02-24 02:13:25 +02:00
|
|
|
IsOAuth2User(ctx context.Context) (bool, error)
|
|
|
|
|
2017-02-21 00:28:38 +02:00
|
|
|
PutUID(ctx context.Context, uid string) error
|
|
|
|
PutProvider(ctx context.Context, provider string) error
|
|
|
|
PutToken(ctx context.Context, token string) error
|
|
|
|
PutRefreshToken(ctx context.Context, refreshToken string) error
|
|
|
|
PutExpiry(ctx context.Context, expiry time.Duration) error
|
|
|
|
|
|
|
|
GetUID(ctx context.Context) (uid string, err error)
|
|
|
|
GetProvider(ctx context.Context) (provider string, err error)
|
|
|
|
GetToken(ctx context.Context) (token string, err error)
|
|
|
|
GetRefreshToken(ctx context.Context) (refreshToken string, err error)
|
|
|
|
GetExpiry(ctx context.Context) (expiry time.Duration, err error)
|
2015-03-14 01:23:43 +02:00
|
|
|
}
|
|
|
|
|
2015-01-19 00:35:44 +02:00
|
|
|
func camelToUnder(in string) string {
|
|
|
|
out := bytes.Buffer{}
|
|
|
|
for i := 0; i < len(in); i++ {
|
|
|
|
chr := in[i]
|
|
|
|
if chr >= 'A' && chr <= 'Z' {
|
|
|
|
if i > 0 {
|
|
|
|
out.WriteByte('_')
|
|
|
|
}
|
|
|
|
out.WriteByte(chr + 'a' - 'A')
|
|
|
|
} else {
|
|
|
|
out.WriteByte(chr)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return out.String()
|
|
|
|
}
|
|
|
|
|
|
|
|
func underToCamel(in string) string {
|
|
|
|
out := bytes.Buffer{}
|
|
|
|
for i := 0; i < len(in); i++ {
|
|
|
|
chr := in[i]
|
|
|
|
|
|
|
|
if first := i == 0; first || chr == '_' {
|
|
|
|
if !first {
|
|
|
|
i++
|
|
|
|
}
|
|
|
|
out.WriteByte(in[i] - 'a' + 'A')
|
|
|
|
} else {
|
|
|
|
out.WriteByte(chr)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return out.String()
|
|
|
|
}
|