Semaphore/api/helpers/helpers.go

132 lines
3.0 KiB
Go
Raw Normal View History

package helpers
2016-01-05 00:32:53 +01:00
import (
"encoding/json"
"errors"
"fmt"
"github.com/ansible-semaphore/semaphore/services/tasks"
2017-02-23 00:21:49 +01:00
"net/http"
2021-10-13 16:33:07 +02:00
"net/url"
"runtime/debug"
2016-01-05 00:32:53 +01:00
"strconv"
"strings"
"github.com/gorilla/context"
log "github.com/sirupsen/logrus"
"github.com/ansible-semaphore/semaphore/db"
"github.com/gorilla/mux"
2016-01-05 00:32:53 +01:00
)
func Store(r *http.Request) db.Store {
return context.Get(r, "store").(db.Store)
}
func TaskPool(r *http.Request) *tasks.TaskPool {
return context.Get(r, "task_pool").(*tasks.TaskPool)
}
2017-02-22 23:17:36 +01:00
func isXHR(w http.ResponseWriter, r *http.Request) bool {
accept := r.Header.Get("Accept")
return !strings.Contains(accept, "text/html")
2016-01-05 00:32:53 +01:00
}
// GetStrParam fetches a parameter from the route variables as an integer
// redirects to a 404 or writes bad request state depending on error state
func GetStrParam(name string, w http.ResponseWriter, r *http.Request) (string, error) {
strParam, ok := mux.Vars(r)[name]
if !ok {
if !isXHR(w, r) {
http.Redirect(w, r, "/404", http.StatusFound)
} else {
w.WriteHeader(http.StatusBadRequest)
}
return "", fmt.Errorf("parameter missed")
}
return strParam, nil
}
// GetIntParam fetches a parameter from the route variables as an integer
// redirects to a 404 or writes bad request state depending on error state
2017-02-22 23:17:36 +01:00
func GetIntParam(name string, w http.ResponseWriter, r *http.Request) (int, error) {
2017-02-23 00:21:49 +01:00
intParam, err := strconv.Atoi(mux.Vars(r)[name])
2016-01-05 00:32:53 +01:00
if err != nil {
if !isXHR(w, r) {
2017-02-23 00:21:49 +01:00
http.Redirect(w, r, "/404", http.StatusFound)
2016-01-05 00:32:53 +01:00
} else {
2017-02-22 23:17:36 +01:00
w.WriteHeader(http.StatusBadRequest)
2016-01-05 00:32:53 +01:00
}
return 0, err
}
return intParam, nil
}
// H just a string-to-anything map
type H map[string]interface{}
// Bind decodes json into object
func Bind(w http.ResponseWriter, r *http.Request, out interface{}) bool {
err := json.NewDecoder(r.Body).Decode(out)
if err != nil {
w.WriteHeader(http.StatusBadRequest)
}
return err == nil
}
// WriteJSON writes object as JSON
func WriteJSON(w http.ResponseWriter, code int, out interface{}) {
w.Header().Set("content-type", "application/json")
w.WriteHeader(code)
if err := json.NewEncoder(w).Encode(out); err != nil {
panic(err)
}
}
func WriteErrorStatus(w http.ResponseWriter, err string, code int) {
WriteJSON(w, code, map[string]string{
"error": err,
})
}
func WriteError(w http.ResponseWriter, err error) {
if errors.Is(err, tasks.ErrInvalidSubscription) {
WriteErrorStatus(w, "You have no subscription.", http.StatusForbidden)
return
}
if errors.Is(err, db.ErrNotFound) {
w.WriteHeader(http.StatusNotFound)
return
}
if errors.Is(err, db.ErrInvalidOperation) {
w.WriteHeader(http.StatusConflict)
return
}
2021-11-02 20:30:45 +01:00
switch e := err.(type) {
case *db.ValidationError:
WriteErrorStatus(w, e.Error(), http.StatusBadRequest)
2021-11-02 20:30:45 +01:00
default:
log.Error(err)
debug.PrintStack()
w.WriteHeader(http.StatusBadRequest)
}
}
2021-10-13 16:33:07 +02:00
func QueryParams(url *url.URL) db.RetrieveQueryParams {
return db.RetrieveQueryParams{
SortBy: url.Query().Get("sort"),
2021-10-13 16:33:07 +02:00
SortInverted: url.Query().Get("order") == "desc",
}
}