1
0
mirror of https://github.com/volatiletech/authboss.git synced 2025-01-10 04:17:59 +02:00

Fix some inconsistencies in clientstate

- Add a ClientStateMiddleware that loads the client state into the
  request context and prepares a ClientStateResponseWriter for
  downstream handlers
- Clean up some of the handling around session and cookie state, for
  example don't write if there are no events to be processed
- Redo the Defaultts() method for config to be useful again.
- Prefix LogoutMethod with Auth to be consistent
This commit is contained in:
Aaron L 2018-02-04 23:28:31 -08:00
parent d4f4f6c443
commit 726204d809
3 changed files with 74 additions and 74 deletions

View File

@ -33,7 +33,7 @@ func (a *Auth) Init(ab *authboss.Authboss) (err error) {
} }
var logoutRouteMethod func(string, http.Handler) var logoutRouteMethod func(string, http.Handler)
switch a.Authboss.Config.Modules.LogoutMethod { switch a.Authboss.Config.Modules.AuthLogoutMethod {
case "GET": case "GET":
logoutRouteMethod = a.Authboss.Config.Core.Router.Get logoutRouteMethod = a.Authboss.Config.Core.Router.Get
case "POST": case "POST":
@ -41,7 +41,7 @@ func (a *Auth) Init(ab *authboss.Authboss) (err error) {
case "DELETE": case "DELETE":
logoutRouteMethod = a.Authboss.Config.Core.Router.Delete logoutRouteMethod = a.Authboss.Config.Core.Router.Delete
default: default:
return errors.Errorf("auth wants to register a logout route but is given an invalid method: %s", a.Authboss.Config.Modules.LogoutMethod) return errors.Errorf("auth wants to register a logout route but is given an invalid method: %s", a.Authboss.Config.Modules.AuthLogoutMethod)
} }
a.Authboss.Config.Core.Router.Get("/login", a.Authboss.Core.ErrorHandler.Wrap(a.LoginGet)) a.Authboss.Config.Core.Router.Get("/login", a.Authboss.Core.ErrorHandler.Wrap(a.LoginGet))

View File

@ -52,7 +52,11 @@ type ClientStateEvent struct {
// There's two major uses for this. To create session storage, and remember me // There's two major uses for this. To create session storage, and remember me
// cookies. // cookies.
type ClientStateReadWriter interface { type ClientStateReadWriter interface {
// ReadState should return a map like structure allowing it to look up
// any values in the current session, or any cookie in the request
ReadState(http.ResponseWriter, *http.Request) (ClientState, error) ReadState(http.ResponseWriter, *http.Request) (ClientState, error)
// WriteState can sometimes be called with a nil ClientState in the event
// that no ClientState was recovered from the request context.
WriteState(http.ResponseWriter, ClientState, []ClientStateEvent) error WriteState(http.ResponseWriter, ClientState, []ClientStateEvent) error
} }
@ -75,20 +79,37 @@ type ClientState interface {
// ClientStateResponseWriter is used to write out the client state at the last // ClientStateResponseWriter is used to write out the client state at the last
// moment before the response code is written. // moment before the response code is written.
type ClientStateResponseWriter struct { type ClientStateResponseWriter struct {
ab *Authboss
http.ResponseWriter http.ResponseWriter
cookieState ClientStateReadWriter
sessionState ClientStateReadWriter
hasWritten bool hasWritten bool
ctx context.Context ctx context.Context
sessionStateEvents []ClientStateEvent sessionStateEvents []ClientStateEvent
cookieStateEvents []ClientStateEvent cookieStateEvents []ClientStateEvent
} }
// ClientStateMiddleware wraps all requests with the ClientStateResponseWriter
func (a *Authboss) ClientStateMiddleware(h http.Handler) http.Handler {
return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
request, err := a.LoadClientState(w, r)
if err != nil {
panic(fmt.Sprintf("failed to load client state: %+v", err))
}
writer := a.NewResponse(w, request)
h.ServeHTTP(writer, request)
})
}
// NewResponse wraps the ResponseWriter with a ClientStateResponseWriter // NewResponse wraps the ResponseWriter with a ClientStateResponseWriter
func (a *Authboss) NewResponse(w http.ResponseWriter, r *http.Request) *ClientStateResponseWriter { func (a *Authboss) NewResponse(w http.ResponseWriter, r *http.Request) *ClientStateResponseWriter {
return &ClientStateResponseWriter{ return &ClientStateResponseWriter{
ab: a,
ResponseWriter: w, ResponseWriter: w,
cookieState: a.Config.Storage.CookieState,
sessionState: a.Config.Storage.SessionState,
ctx: r.Context(), ctx: r.Context(),
} }
} }
@ -174,24 +195,32 @@ func (c *ClientStateResponseWriter) putClientState() error {
} }
c.hasWritten = true c.hasWritten = true
sessionStateIntf := c.ctx.Value(ctxKeySessionState) if len(c.cookieStateEvents) == 0 && len(c.sessionStateEvents) == 0 {
cookieStateIntf := c.ctx.Value(ctxKeyCookieState) return nil
var session, cookie ClientState
if sessionStateIntf != nil {
session = sessionStateIntf.(ClientState)
}
if cookieStateIntf != nil {
cookie = cookieStateIntf.(ClientState)
} }
if c.ab.Storage.SessionState != nil { if c.sessionState != nil && len(c.sessionStateEvents) > 0 {
err := c.ab.Storage.SessionState.WriteState(c, session, c.sessionStateEvents) sessionStateIntf := c.ctx.Value(ctxKeySessionState)
var session ClientState
if sessionStateIntf != nil {
session = sessionStateIntf.(ClientState)
}
err := c.sessionState.WriteState(c, session, c.sessionStateEvents)
if err != nil { if err != nil {
return err return err
} }
} }
if c.ab.Storage.CookieState != nil { if c.cookieState != nil && len(c.cookieStateEvents) > 0 {
err := c.ab.Storage.CookieState.WriteState(c, cookie, c.cookieStateEvents) cookieStateIntf := c.ctx.Value(ctxKeyCookieState)
var cookie ClientState
if cookieStateIntf != nil {
cookie = cookieStateIntf.(ClientState)
}
err := c.cookieState.WriteState(c, cookie, c.cookieStateEvents)
if err != nil { if err != nil {
return err return err
} }

View File

@ -2,6 +2,8 @@ package authboss
import ( import (
"time" "time"
"golang.org/x/crypto/bcrypt"
) )
// Config holds all the configuration for both authboss and it's modules. // Config holds all the configuration for both authboss and it's modules.
@ -29,31 +31,31 @@ type Config struct {
// BCryptCost is the cost of the bcrypt password hashing function. // BCryptCost is the cost of the bcrypt password hashing function.
AuthBCryptCost int AuthBCryptCost int
// LogoutMethod is the method the logout route should use (default should be DELETE) // AuthLogoutMethod is the method the logout route should use (default should be DELETE)
LogoutMethod string AuthLogoutMethod string
// OAuth2Providers lists all providers that can be used. See
// OAuthProvider documentation for more details.
OAuth2Providers map[string]OAuth2Provider
// PreserveFields are fields used with registration that are to be rendered when
// post fails.
PreserveFields []string
// ExpireAfter controls the time an account is idle before being logged out // ExpireAfter controls the time an account is idle before being logged out
// by the ExpireMiddleware. // by the ExpireMiddleware.
ExpireAfter time.Duration ExpireAfter time.Duration
// RecoverTokenDuration controls how long a token sent via email for password
// recovery is valid for.
RecoverTokenDuration time.Duration
// LockAfter this many tries. // LockAfter this many tries.
LockAfter int LockAfter int
// LockWindow is the waiting time before the number of attemps are reset. // LockWindow is the waiting time before the number of attemps are reset.
LockWindow time.Duration LockWindow time.Duration
// LockDuration is how long an account is locked for. // LockDuration is how long an account is locked for.
LockDuration time.Duration LockDuration time.Duration
// RegisterPreserveFields are fields used with registration that are to be rendered when
// post fails.
RegisterPreserveFields []string
// RecoverTokenDuration controls how long a token sent via email for password
// recovery is valid for.
RecoverTokenDuration time.Duration
// OAuth2Providers lists all providers that can be used. See
// OAuthProvider documentation for more details.
OAuth2Providers map[string]OAuth2Provider
} }
Mail struct { Mail struct {
@ -117,49 +119,18 @@ type Config struct {
// Defaults sets the configuration's default values. // Defaults sets the configuration's default values.
func (c *Config) Defaults() { func (c *Config) Defaults() {
/*c.MountPath = "/" c.Paths.Mount = "/"
c.ViewsPath = "./" c.Paths.RootURL = "http://localhost:8080"
c.RootURL = "http://localhost:8080" c.Paths.AuthLoginOK = "/"
c.BCryptCost = bcrypt.DefaultCost c.Paths.AuthLogoutOK = "/"
c.Paths.RecoverOK = "/"
c.Paths.RegisterOK = "/"
c.PrimaryID = StoreEmail c.Modules.AuthBCryptCost = bcrypt.DefaultCost
c.Modules.AuthLogoutMethod = "DELETE"
c.AuthLoginOKPath = "/" c.Modules.ExpireAfter = 60 * time.Minute
c.AuthLoginFailPath = "/" c.Modules.LockAfter = 3
c.AuthLogoutOKPath = "/" c.Modules.LockWindow = 5 * time.Minute
c.Modules.LockDuration = 5 * time.Hour
c.RecoverOKPath = "/" c.Modules.RecoverTokenDuration = time.Duration(24) * time.Hour
c.RecoverTokenDuration = time.Duration(24) * time.Hour
c.RegisterOKPath = "/"
c.Policies = []Validator{
Rules{
FieldName: "email",
Required: true,
AllowWhitespace: false,
},
Rules{
FieldName: "password",
Required: true,
MinLength: 4,
MaxLength: 8,
AllowWhitespace: false,
},
}
c.ConfirmFields = []string{
StorePassword, ConfirmPrefix + StorePassword,
}
c.ExpireAfter = 60 * time.Minute
c.LockAfter = 3
c.LockWindow = 5 * time.Minute
c.LockDuration = 5 * time.Hour
c.LogWriter = NewDefaultLogger()
c.Mailer = LogMailer(ioutil.Discard)
c.ContextProvider = func(req *http.Request) context.Context {
return context.TODO()
}*/
} }