2018-04-10 15:20:23 +02:00
|
|
|
package app
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2018-05-19 20:58:57 +02:00
|
|
|
"fmt"
|
2018-05-23 21:40:30 +02:00
|
|
|
"github.com/pkg/errors"
|
2018-05-19 20:58:57 +02:00
|
|
|
log "github.com/sirupsen/logrus"
|
2018-05-21 21:38:16 +02:00
|
|
|
"golang.org/x/crypto/bcrypt"
|
|
|
|
"net/http"
|
2018-04-10 15:20:23 +02:00
|
|
|
)
|
|
|
|
|
2018-05-19 20:58:57 +02:00
|
|
|
type contextKey int
|
2018-04-10 15:20:23 +02:00
|
|
|
|
2018-05-21 21:38:16 +02:00
|
|
|
var authInfoKey contextKey
|
|
|
|
|
|
|
|
// AuthInfo holds the username and authentication status
|
2018-05-19 20:58:57 +02:00
|
|
|
type AuthInfo struct {
|
|
|
|
Username string
|
|
|
|
Authenticated bool
|
2018-04-10 15:20:23 +02:00
|
|
|
}
|
|
|
|
|
2018-05-21 21:38:16 +02:00
|
|
|
// authWebdavHandlerFunc is a type definition which holds a context and application reference to
|
2018-04-10 15:20:23 +02:00
|
|
|
// match the AuthWebdavHandler interface.
|
2018-05-19 20:58:57 +02:00
|
|
|
type authWebdavHandlerFunc func(c context.Context, w http.ResponseWriter, r *http.Request, a *App)
|
2018-04-10 15:20:23 +02:00
|
|
|
|
|
|
|
// ServeHTTP simply calls the AuthWebdavHandlerFunc with given parameters
|
2018-05-19 20:58:57 +02:00
|
|
|
func (f authWebdavHandlerFunc) ServeHTTP(c context.Context, w http.ResponseWriter, r *http.Request, a *App) {
|
2018-04-10 15:20:23 +02:00
|
|
|
f(c, w, r, a)
|
|
|
|
}
|
|
|
|
|
2018-05-21 21:38:16 +02:00
|
|
|
// NewBasicAuthWebdavHandler creates a new http handler with basic auth features.
|
|
|
|
// The handler will use the application config for user and password lookups.
|
|
|
|
func NewBasicAuthWebdavHandler(a *App) http.Handler {
|
|
|
|
return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
|
|
ctx := context.Background()
|
|
|
|
handlerFunc := authWebdavHandlerFunc(handle)
|
|
|
|
handlerFunc.ServeHTTP(ctx, w, r, a)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2018-05-23 21:40:30 +02:00
|
|
|
func authenticate(config *Config, username, password string) (*AuthInfo, error) {
|
2018-06-17 14:14:55 +02:00
|
|
|
if !config.AuthenticationNeeded() {
|
2018-06-17 14:27:40 +02:00
|
|
|
return &AuthInfo{Username: "", Authenticated: false}, nil
|
2018-06-17 14:14:55 +02:00
|
|
|
}
|
|
|
|
|
2018-05-19 20:58:57 +02:00
|
|
|
if username == "" || password == "" {
|
2018-05-23 21:40:30 +02:00
|
|
|
return &AuthInfo{Username: username, Authenticated: false}, errors.New("username not found or password empty")
|
2018-05-19 20:58:57 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
user := config.Users[username]
|
|
|
|
if user == nil {
|
2018-05-23 21:40:30 +02:00
|
|
|
return &AuthInfo{Username: username, Authenticated: false}, errors.New("user not found")
|
2018-05-19 20:58:57 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
err := bcrypt.CompareHashAndPassword([]byte(user.Password), []byte(password))
|
|
|
|
if err != nil {
|
2018-05-23 21:40:30 +02:00
|
|
|
return &AuthInfo{Username: username, Authenticated: false}, errors.New("Password doesn't match")
|
2018-05-19 20:58:57 +02:00
|
|
|
}
|
|
|
|
|
2018-05-23 21:40:30 +02:00
|
|
|
return &AuthInfo{Username: username, Authenticated: true}, nil
|
2018-05-19 20:58:57 +02:00
|
|
|
}
|
|
|
|
|
2018-05-21 21:38:16 +02:00
|
|
|
// AuthFromContext returns information about the authentication state of the current user.
|
2018-05-19 20:58:57 +02:00
|
|
|
func AuthFromContext(ctx context.Context) *AuthInfo {
|
|
|
|
info, ok := ctx.Value(authInfoKey).(*AuthInfo)
|
|
|
|
if !ok {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return info
|
|
|
|
}
|
|
|
|
|
|
|
|
func handle(ctx context.Context, w http.ResponseWriter, r *http.Request, a *App) {
|
2018-06-17 14:14:55 +02:00
|
|
|
// if there are no users, we don't need authentication here
|
|
|
|
if (!a.Config.AuthenticationNeeded()) {
|
|
|
|
a.Handler.ServeHTTP(w, r.WithContext(ctx))
|
|
|
|
return
|
|
|
|
}
|
2018-05-19 20:58:57 +02:00
|
|
|
|
2018-06-17 14:14:55 +02:00
|
|
|
username, password, ok := httpAuth(r, a.Config)
|
2018-05-19 20:58:57 +02:00
|
|
|
if !ok {
|
|
|
|
writeUnauthorized(w, a.Config.Realm)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2018-05-23 21:40:30 +02:00
|
|
|
authInfo, err := authenticate(a.Config, username, password)
|
|
|
|
if err != nil {
|
|
|
|
log.WithField("user", username).Warn(err.Error())
|
|
|
|
}
|
|
|
|
|
2018-05-19 20:58:57 +02:00
|
|
|
if !authInfo.Authenticated {
|
|
|
|
writeUnauthorized(w, a.Config.Realm)
|
2018-04-10 15:20:23 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2018-05-19 20:58:57 +02:00
|
|
|
ctx = context.WithValue(ctx, authInfoKey, authInfo)
|
2018-04-10 22:02:48 +02:00
|
|
|
a.Handler.ServeHTTP(w, r.WithContext(ctx))
|
2018-04-10 15:20:23 +02:00
|
|
|
}
|
|
|
|
|
2018-06-17 14:14:55 +02:00
|
|
|
func httpAuth(r *http.Request, config *Config) (string, string, bool) {
|
|
|
|
if config.AuthenticationNeeded() {
|
|
|
|
username, password, ok := r.BasicAuth()
|
|
|
|
return username, password, ok
|
|
|
|
}
|
|
|
|
|
|
|
|
return "", "", true
|
|
|
|
}
|
|
|
|
|
2018-05-19 20:58:57 +02:00
|
|
|
func writeUnauthorized(w http.ResponseWriter, realm string) {
|
2018-05-21 21:38:16 +02:00
|
|
|
w.Header().Set("WWW-Authenticate", "Basic realm="+realm)
|
2018-05-19 20:58:57 +02:00
|
|
|
w.WriteHeader(http.StatusUnauthorized)
|
|
|
|
w.Write([]byte(fmt.Sprintf("%d %s", http.StatusUnauthorized, "Unauthorized")))
|
2018-04-10 15:20:23 +02:00
|
|
|
}
|
2018-05-23 21:40:30 +02:00
|
|
|
|
2018-05-24 21:37:42 +02:00
|
|
|
// GenHash generates a bcrypt hashed password string
|
2018-05-23 21:40:30 +02:00
|
|
|
func GenHash(password []byte) string {
|
|
|
|
pw, err := bcrypt.GenerateFromPassword(password, 10)
|
|
|
|
if err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return string(pw)
|
|
|
|
}
|