Semaphore/api/auth.go

125 lines
2.8 KiB
Go
Raw Normal View History

2016-05-24 11:55:48 +02:00
package api
2016-01-05 00:32:53 +01:00
import (
log "github.com/Sirupsen/logrus"
"github.com/ansible-semaphore/semaphore/api/helpers"
"github.com/ansible-semaphore/semaphore/db"
"github.com/ansible-semaphore/semaphore/util"
"github.com/gorilla/context"
2017-02-23 00:21:49 +01:00
"net/http"
"strings"
2016-01-05 00:32:53 +01:00
"time"
)
func authenticationHandler(w http.ResponseWriter, r *http.Request) {
var userID int
authHeader := strings.ToLower(r.Header.Get("authorization"))
if len(authHeader) > 0 && strings.Contains(authHeader, "bearer") {
token, err := helpers.Store(r).GetAPIToken(strings.Replace(authHeader, "bearer ", "", 1))
if err != nil {
if err != db.ErrNotFound {
log.Error(err)
}
w.WriteHeader(http.StatusUnauthorized)
return
}
2016-01-05 00:32:53 +01:00
userID = token.UserID
} else {
// fetch session from cookie
cookie, err := r.Cookie("semaphore")
if err != nil {
w.WriteHeader(http.StatusUnauthorized)
return
}
2016-01-05 00:32:53 +01:00
value := make(map[string]interface{})
if err = util.Cookie.Decode("semaphore", cookie.Value, &value); err != nil {
w.WriteHeader(http.StatusUnauthorized)
return
}
2016-01-05 00:32:53 +01:00
user, ok := value["user"]
sessionVal, okSession := value["session"]
if !ok || !okSession {
w.WriteHeader(http.StatusUnauthorized)
return
}
userID = user.(int)
sessionID := sessionVal.(int)
2016-01-05 00:32:53 +01:00
// fetch session
session, err := helpers.Store(r).GetSession(userID, sessionID)
2016-01-05 00:32:53 +01:00
if err != nil {
w.WriteHeader(http.StatusUnauthorized)
return
}
2016-01-05 00:32:53 +01:00
if time.Since(session.LastActive).Hours() > 7*24 {
// more than week old unused session
// destroy.
if err := helpers.Store(r).ExpireSession(userID, sessionID); err != nil {
// it is internal error, it doesn't concern the user
log.Error(err)
}
w.WriteHeader(http.StatusUnauthorized)
return
}
2016-01-05 00:32:53 +01:00
if err := helpers.Store(r).TouchSession(userID, sessionID); err != nil {
log.Error(err)
w.WriteHeader(http.StatusUnauthorized)
return
}
}
user, err := helpers.Store(r).GetUser(userID)
if err != nil {
if err != db.ErrNotFound {
// internal error
log.Error(err)
}
w.WriteHeader(http.StatusUnauthorized)
return
}
if util.Config.DemoMode {
if !user.Admin && r.Method != "GET" &&
!strings.HasSuffix(r.URL.Path, "/tasks") &&
!strings.HasSuffix(r.URL.Path, "/stop") {
w.WriteHeader(http.StatusUnauthorized)
return
2022-01-18 23:19:31 +01:00
}
}
2022-01-18 23:19:31 +01:00
context.Set(r, "user", &user)
}
// nolint: gocyclo
func authentication(next http.Handler) http.Handler {
return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
authenticationHandler(w, r)
next.ServeHTTP(w, r)
})
}
// nolint: gocyclo
func authenticationWithStore(next http.Handler) http.Handler {
return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
store := helpers.Store(r)
db.StoreSession(store, r.URL.String(), func() {
authenticationHandler(w, r)
})
next.ServeHTTP(w, r)
})
2016-01-05 00:32:53 +01:00
}