2017-05-23 18:56:10 +00:00
|
|
|
package handler
|
2016-12-18 05:21:29 +00:00
|
|
|
|
|
|
|
import (
|
|
|
|
"github.com/portainer/portainer"
|
|
|
|
|
|
|
|
"encoding/json"
|
|
|
|
"log"
|
|
|
|
"net/http"
|
|
|
|
"os"
|
2016-12-25 20:34:02 +00:00
|
|
|
|
|
|
|
"github.com/asaskevich/govalidator"
|
|
|
|
"github.com/gorilla/mux"
|
2017-05-23 18:56:10 +00:00
|
|
|
httperror "github.com/portainer/portainer/http/error"
|
|
|
|
"github.com/portainer/portainer/http/security"
|
2016-12-18 05:21:29 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// AuthHandler represents an HTTP API handler for managing authentication.
|
|
|
|
type AuthHandler struct {
|
|
|
|
*mux.Router
|
2017-08-10 08:35:23 +00:00
|
|
|
Logger *log.Logger
|
|
|
|
authDisabled bool
|
|
|
|
UserService portainer.UserService
|
|
|
|
CryptoService portainer.CryptoService
|
|
|
|
JWTService portainer.JWTService
|
|
|
|
LDAPService portainer.LDAPService
|
|
|
|
SettingsService portainer.SettingsService
|
2016-12-18 05:21:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
const (
|
|
|
|
// ErrInvalidCredentialsFormat is an error raised when credentials format is not valid
|
|
|
|
ErrInvalidCredentialsFormat = portainer.Error("Invalid credentials format")
|
|
|
|
// ErrInvalidCredentials is an error raised when credentials for a user are invalid
|
|
|
|
ErrInvalidCredentials = portainer.Error("Invalid credentials")
|
2017-02-01 09:13:48 +00:00
|
|
|
// ErrAuthDisabled is an error raised when trying to access the authentication endpoints
|
|
|
|
// when the server has been started with the --no-auth flag
|
|
|
|
ErrAuthDisabled = portainer.Error("Authentication is disabled")
|
2016-12-18 05:21:29 +00:00
|
|
|
)
|
|
|
|
|
2016-12-25 20:34:02 +00:00
|
|
|
// NewAuthHandler returns a new instance of AuthHandler.
|
2017-05-23 18:56:10 +00:00
|
|
|
func NewAuthHandler(bouncer *security.RequestBouncer, authDisabled bool) *AuthHandler {
|
2016-12-18 05:21:29 +00:00
|
|
|
h := &AuthHandler{
|
2017-05-23 18:56:10 +00:00
|
|
|
Router: mux.NewRouter(),
|
|
|
|
Logger: log.New(os.Stderr, "", log.LstdFlags),
|
|
|
|
authDisabled: authDisabled,
|
2016-12-18 05:21:29 +00:00
|
|
|
}
|
2017-03-12 16:24:15 +00:00
|
|
|
h.Handle("/auth",
|
2017-05-23 18:56:10 +00:00
|
|
|
bouncer.PublicAccess(http.HandlerFunc(h.handlePostAuth)))
|
2017-03-12 16:24:15 +00:00
|
|
|
|
2016-12-18 05:21:29 +00:00
|
|
|
return h
|
|
|
|
}
|
|
|
|
|
|
|
|
func (handler *AuthHandler) handlePostAuth(w http.ResponseWriter, r *http.Request) {
|
2016-12-25 20:34:02 +00:00
|
|
|
if r.Method != http.MethodPost {
|
2017-05-23 18:56:10 +00:00
|
|
|
httperror.WriteMethodNotAllowedResponse(w, []string{http.MethodPost})
|
2016-12-18 05:21:29 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2017-02-01 09:13:48 +00:00
|
|
|
if handler.authDisabled {
|
2017-05-23 18:56:10 +00:00
|
|
|
httperror.WriteErrorResponse(w, ErrAuthDisabled, http.StatusServiceUnavailable, handler.Logger)
|
2017-02-01 09:13:48 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2016-12-18 05:21:29 +00:00
|
|
|
var req postAuthRequest
|
|
|
|
if err := json.NewDecoder(r.Body).Decode(&req); err != nil {
|
2017-05-23 18:56:10 +00:00
|
|
|
httperror.WriteErrorResponse(w, ErrInvalidJSON, http.StatusBadRequest, handler.Logger)
|
2016-12-18 05:21:29 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
_, err := govalidator.ValidateStruct(req)
|
|
|
|
if err != nil {
|
2017-05-23 18:56:10 +00:00
|
|
|
httperror.WriteErrorResponse(w, ErrInvalidCredentialsFormat, http.StatusBadRequest, handler.Logger)
|
2016-12-18 05:21:29 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
var username = req.Username
|
|
|
|
var password = req.Password
|
|
|
|
|
2017-03-12 16:24:15 +00:00
|
|
|
u, err := handler.UserService.UserByUsername(username)
|
2016-12-18 05:21:29 +00:00
|
|
|
if err == portainer.ErrUserNotFound {
|
2017-07-12 15:22:14 +00:00
|
|
|
httperror.WriteErrorResponse(w, ErrInvalidCredentials, http.StatusBadRequest, handler.Logger)
|
2016-12-18 05:21:29 +00:00
|
|
|
return
|
|
|
|
} else if err != nil {
|
2017-05-23 18:56:10 +00:00
|
|
|
httperror.WriteErrorResponse(w, err, http.StatusInternalServerError, handler.Logger)
|
2016-12-18 05:21:29 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2017-08-10 08:35:23 +00:00
|
|
|
settings, err := handler.SettingsService.Settings()
|
2016-12-18 05:21:29 +00:00
|
|
|
if err != nil {
|
2017-08-10 08:35:23 +00:00
|
|
|
httperror.WriteErrorResponse(w, err, http.StatusInternalServerError, handler.Logger)
|
2016-12-18 05:21:29 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2017-08-10 08:35:23 +00:00
|
|
|
if settings.AuthenticationMethod == portainer.AuthenticationLDAP && u.ID != 1 {
|
|
|
|
err = handler.LDAPService.AuthenticateUser(username, password, &settings.LDAPSettings)
|
|
|
|
if err != nil {
|
|
|
|
httperror.WriteErrorResponse(w, err, http.StatusInternalServerError, handler.Logger)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
err = handler.CryptoService.CompareHashAndData(u.Password, password)
|
|
|
|
if err != nil {
|
|
|
|
httperror.WriteErrorResponse(w, ErrInvalidCredentials, http.StatusUnprocessableEntity, handler.Logger)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-12-18 05:21:29 +00:00
|
|
|
tokenData := &portainer.TokenData{
|
2017-03-12 16:24:15 +00:00
|
|
|
ID: u.ID,
|
|
|
|
Username: u.Username,
|
|
|
|
Role: u.Role,
|
2016-12-18 05:21:29 +00:00
|
|
|
}
|
2017-08-10 08:35:23 +00:00
|
|
|
|
2016-12-18 05:21:29 +00:00
|
|
|
token, err := handler.JWTService.GenerateToken(tokenData)
|
|
|
|
if err != nil {
|
2017-05-23 18:56:10 +00:00
|
|
|
httperror.WriteErrorResponse(w, err, http.StatusInternalServerError, handler.Logger)
|
2016-12-18 05:21:29 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
encodeJSON(w, &postAuthResponse{JWT: token}, handler.Logger)
|
|
|
|
}
|
|
|
|
|
|
|
|
type postAuthRequest struct {
|
2017-05-23 18:56:10 +00:00
|
|
|
Username string `valid:"required"`
|
2016-12-18 05:21:29 +00:00
|
|
|
Password string `valid:"required"`
|
|
|
|
}
|
|
|
|
|
|
|
|
type postAuthResponse struct {
|
|
|
|
JWT string `json:"jwt"`
|
|
|
|
}
|