2015-03-15 17:06:08 +02:00
|
|
|
// Package auth implements password based user logins.
|
2015-01-04 20:33:53 +02:00
|
|
|
package auth
|
|
|
|
|
|
|
|
import (
|
2015-01-15 23:24:12 +02:00
|
|
|
"fmt"
|
2015-01-04 20:33:53 +02:00
|
|
|
"net/http"
|
|
|
|
|
2017-02-22 01:04:30 +02:00
|
|
|
"github.com/pkg/errors"
|
2017-07-31 04:39:33 +02:00
|
|
|
"github.com/volatiletech/authboss"
|
|
|
|
"github.com/volatiletech/authboss/internal/response"
|
2017-02-22 01:04:30 +02:00
|
|
|
"golang.org/x/crypto/bcrypt"
|
2015-01-04 20:33:53 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
2015-02-26 22:52:45 +02:00
|
|
|
tplLogin = "login.html.tpl"
|
2015-01-13 07:08:52 +02:00
|
|
|
)
|
2015-01-11 08:49:06 +02:00
|
|
|
|
2015-01-04 20:33:53 +02:00
|
|
|
func init() {
|
2016-05-09 19:20:10 +02:00
|
|
|
authboss.RegisterModule("auth", &Auth{})
|
2015-01-04 20:33:53 +02:00
|
|
|
}
|
|
|
|
|
2015-03-16 23:42:45 +02:00
|
|
|
// Auth module
|
2015-02-24 01:51:42 +02:00
|
|
|
type Auth struct {
|
2015-04-01 00:27:47 +02:00
|
|
|
*authboss.Authboss
|
2015-01-04 20:33:53 +02:00
|
|
|
}
|
|
|
|
|
2018-02-02 03:23:31 +02:00
|
|
|
// Init module
|
|
|
|
func (a *Auth) Init(ab *authboss.Authboss) (err error) {
|
2015-04-01 00:27:47 +02:00
|
|
|
a.Authboss = ab
|
|
|
|
|
2018-02-02 03:23:31 +02:00
|
|
|
if err := a.Authboss.Config.Core.ViewRenderer.Load(tplLogin); err != nil {
|
|
|
|
return err
|
2015-02-24 01:51:42 +02:00
|
|
|
}
|
|
|
|
|
2018-02-02 03:23:31 +02:00
|
|
|
var logoutRouteMethod func(string, http.Handler)
|
|
|
|
switch a.Authboss.Config.Modules.LogoutMethod {
|
|
|
|
case "GET":
|
|
|
|
logoutRouteMethod = a.Authboss.Config.Core.Router.Get
|
|
|
|
case "POST":
|
|
|
|
logoutRouteMethod = a.Authboss.Config.Core.Router.Post
|
|
|
|
case "DELETE":
|
|
|
|
logoutRouteMethod = a.Authboss.Config.Core.Router.Delete
|
|
|
|
default:
|
|
|
|
return errors.Errorf("auth wants to register a logout route but is given an invalid method: %s", a.Authboss.Config.Modules.LogoutMethod)
|
2015-02-25 01:01:56 +02:00
|
|
|
}
|
|
|
|
|
2018-02-02 03:23:31 +02:00
|
|
|
a.Authboss.Config.Core.Router.Get("/login", http.HandlerFunc(loginGet))
|
|
|
|
a.Authboss.Config.Core.Router.Post("/login", http.HandlerFunc(loginPost))
|
|
|
|
logoutRouteMethod("/logout", http.HandlerFunc(logout))
|
2015-01-04 20:33:53 +02:00
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2018-02-02 03:23:31 +02:00
|
|
|
func (a *Auth) loginGet(w http.ResponseWriter, r *http.Request) error {
|
|
|
|
data := authboss.NewHTMLData(
|
|
|
|
"showRemember", a.IsLoaded("remember"),
|
|
|
|
"showRecover", a.IsLoaded("recover"),
|
|
|
|
"showRegister", a.IsLoaded("register"),
|
|
|
|
"primaryID", a.PrimaryID,
|
|
|
|
"primaryIDValue", "",
|
|
|
|
)
|
|
|
|
return a.templates.Render(ctx, w, r, tplLogin, data)
|
2015-01-04 20:33:53 +02:00
|
|
|
}
|
|
|
|
|
2018-02-02 03:23:31 +02:00
|
|
|
func (a *Auth) loginPost(w http.ResponseWriter, r *http.Request) error {
|
2015-01-04 20:33:53 +02:00
|
|
|
switch r.Method {
|
|
|
|
case methodGET:
|
|
|
|
case methodPOST:
|
2015-08-02 20:52:23 +02:00
|
|
|
key := r.FormValue(a.PrimaryID)
|
|
|
|
password := r.FormValue("password")
|
2015-01-16 00:01:01 +02:00
|
|
|
|
2015-02-21 09:33:35 +02:00
|
|
|
errData := authboss.NewHTMLData(
|
2015-04-01 00:27:47 +02:00
|
|
|
"error", fmt.Sprintf("invalid %s and/or password", a.PrimaryID),
|
|
|
|
"primaryID", a.PrimaryID,
|
2015-02-24 01:51:42 +02:00
|
|
|
"primaryIDValue", key,
|
2015-04-01 00:27:47 +02:00
|
|
|
"showRemember", a.IsLoaded("remember"),
|
|
|
|
"showRecover", a.IsLoaded("recover"),
|
2015-04-10 21:04:26 +02:00
|
|
|
"showRegister", a.IsLoaded("register"),
|
2015-02-21 09:33:35 +02:00
|
|
|
)
|
|
|
|
|
2015-04-03 20:49:28 +02:00
|
|
|
if valid, err := validateCredentials(ctx, key, password); err != nil {
|
|
|
|
errData["error"] = "Internal server error"
|
2016-05-07 08:12:20 +02:00
|
|
|
fmt.Fprintf(ctx.LogWriter, "auth: validate credentials failed: %v\n", err)
|
2015-04-03 20:49:28 +02:00
|
|
|
return a.templates.Render(ctx, w, r, tplLogin, errData)
|
|
|
|
} else if !valid {
|
2018-02-02 02:31:08 +02:00
|
|
|
if err := a.Events.FireAfter(authboss.EventAuthFail, ctx); err != nil {
|
2016-05-07 08:12:20 +02:00
|
|
|
fmt.Fprintf(ctx.LogWriter, "EventAuthFail callback error'd out: %v\n", err)
|
2015-08-02 18:52:30 +02:00
|
|
|
}
|
2015-02-21 09:33:35 +02:00
|
|
|
return a.templates.Render(ctx, w, r, tplLogin, errData)
|
2015-01-11 09:12:40 +02:00
|
|
|
}
|
2015-01-16 01:10:47 +02:00
|
|
|
|
2018-02-02 02:31:08 +02:00
|
|
|
interrupted, err := a.Events.FireBefore(authboss.EventAuth, ctx)
|
2015-02-26 09:05:14 +02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
} else if interrupted != authboss.InterruptNone {
|
|
|
|
var reason string
|
|
|
|
switch interrupted {
|
|
|
|
case authboss.InterruptAccountLocked:
|
|
|
|
reason = "Your account has been locked."
|
|
|
|
case authboss.InterruptAccountNotConfirmed:
|
|
|
|
reason = "Your account has not been confirmed."
|
|
|
|
}
|
2015-04-01 00:27:47 +02:00
|
|
|
response.Redirect(ctx, w, r, a.AuthLoginFailPath, "", reason, false)
|
2015-02-26 09:05:14 +02:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
ctx.SessionStorer.Put(authboss.SessionKey, key)
|
|
|
|
ctx.SessionStorer.Del(authboss.SessionHalfAuthKey)
|
2015-08-02 23:02:14 +02:00
|
|
|
ctx.Values = map[string]string{authboss.CookieRemember: r.FormValue(authboss.CookieRemember)}
|
2015-02-26 09:05:14 +02:00
|
|
|
|
2018-02-02 02:31:08 +02:00
|
|
|
if err := a.Events.FireAfter(authboss.EventAuth, ctx); err != nil {
|
2015-02-26 09:20:02 +02:00
|
|
|
return err
|
|
|
|
}
|
2015-04-01 00:27:47 +02:00
|
|
|
response.Redirect(ctx, w, r, a.AuthLoginOKPath, "", "", true)
|
2015-01-04 20:33:53 +02:00
|
|
|
default:
|
|
|
|
w.WriteHeader(http.StatusMethodNotAllowed)
|
|
|
|
}
|
2015-02-21 09:33:35 +02:00
|
|
|
|
|
|
|
return nil
|
2015-01-04 20:33:53 +02:00
|
|
|
}
|
|
|
|
|
2018-02-02 03:23:31 +02:00
|
|
|
func validateCredentials(key, password string) (bool, error) {
|
2015-07-02 03:07:26 +02:00
|
|
|
if err := ctx.LoadUser(key); err == authboss.ErrUserNotFound {
|
|
|
|
return false, nil
|
|
|
|
} else if err != nil {
|
2015-04-03 20:49:28 +02:00
|
|
|
return false, err
|
2015-01-15 23:24:12 +02:00
|
|
|
}
|
|
|
|
|
2015-02-22 23:16:11 +02:00
|
|
|
actualPassword, err := ctx.User.StringErr(authboss.StorePassword)
|
2015-02-21 09:33:35 +02:00
|
|
|
if err != nil {
|
2015-04-03 20:49:28 +02:00
|
|
|
return false, err
|
2015-01-15 23:24:12 +02:00
|
|
|
}
|
|
|
|
|
2015-02-21 09:33:35 +02:00
|
|
|
if err := bcrypt.CompareHashAndPassword([]byte(actualPassword), []byte(password)); err != nil {
|
2015-04-03 20:49:28 +02:00
|
|
|
return false, nil
|
2015-01-11 08:49:06 +02:00
|
|
|
}
|
|
|
|
|
2015-04-03 20:49:28 +02:00
|
|
|
return true, nil
|
2015-01-11 08:49:06 +02:00
|
|
|
}
|
|
|
|
|
2018-02-02 03:23:31 +02:00
|
|
|
func (a *Auth) logout(w http.ResponseWriter, r *http.Request) error {
|
|
|
|
ctx.SessionStorer.Del(authboss.SessionKey)
|
|
|
|
ctx.CookieStorer.Del(authboss.CookieRemember)
|
|
|
|
ctx.SessionStorer.Del(authboss.SessionLastAction)
|
2015-03-03 08:09:32 +02:00
|
|
|
|
2018-02-02 03:23:31 +02:00
|
|
|
response.Redirect(ctx, w, r, a.AuthLogoutOKPath, "You have logged out", "", true)
|
2015-02-21 09:33:35 +02:00
|
|
|
|
|
|
|
return nil
|
2015-01-04 20:33:53 +02:00
|
|
|
}
|