mirror of
https://github.com/bitechdev/ResolveSpec.git
synced 2026-05-21 19:45:33 +00:00
* Implement SetAuthenticateCallback in authenticators * Update Authenticate methods to use callback on failure
58 lines
1.9 KiB
Go
58 lines
1.9 KiB
Go
package security
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"net/http"
|
|
)
|
|
|
|
// ChainAuthenticator tries each authenticator in order, returning the first success.
|
|
// Login and Logout are delegated to the primary authenticator.
|
|
type ChainAuthenticator struct {
|
|
authenticators []Authenticator
|
|
authenticateCallback func(r *http.Request) (*UserContext, error)
|
|
}
|
|
|
|
// NewChainAuthenticator creates a ChainAuthenticator from the given authenticators.
|
|
// At least one authenticator is required; the first is treated as primary for Login/Logout.
|
|
func NewChainAuthenticator(primary Authenticator, rest ...Authenticator) *ChainAuthenticator {
|
|
return &ChainAuthenticator{
|
|
authenticators: append([]Authenticator{primary}, rest...),
|
|
}
|
|
}
|
|
|
|
func (c *ChainAuthenticator) Authenticate(r *http.Request) (*UserContext, error) {
|
|
var lastErr error
|
|
for _, a := range c.authenticators {
|
|
if uc, err := a.Authenticate(r); err == nil {
|
|
return uc, nil
|
|
} else {
|
|
lastErr = err
|
|
}
|
|
}
|
|
if c.authenticateCallback != nil {
|
|
return c.authenticateCallback(r)
|
|
}
|
|
return nil, fmt.Errorf("all authenticators failed; last error: %w", lastErr)
|
|
}
|
|
|
|
func (c *ChainAuthenticator) SetAuthenticateCallback(fn func(r *http.Request) (*UserContext, error)) {
|
|
c.authenticateCallback = fn
|
|
}
|
|
|
|
func (c *ChainAuthenticator) Login(ctx context.Context, req LoginRequest) (*LoginResponse, error) {
|
|
return c.authenticators[0].Login(ctx, req)
|
|
}
|
|
|
|
func (c *ChainAuthenticator) LoginWithCookie(ctx context.Context, req LoginRequest, w http.ResponseWriter) (*LoginResponse, error) {
|
|
return c.authenticators[0].LoginWithCookie(ctx, req, w)
|
|
}
|
|
|
|
func (c *ChainAuthenticator) Logout(ctx context.Context, req LogoutRequest) error {
|
|
return c.authenticators[0].Logout(ctx, req)
|
|
}
|
|
|
|
func (c *ChainAuthenticator) LogoutWithCookie(ctx context.Context, req LogoutRequest, w http.ResponseWriter) error {
|
|
return c.authenticators[0].LogoutWithCookie(ctx, req, w)
|
|
}
|