2020-10-18 15:35:47 +02:00
|
|
|
package controller
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/json"
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
"net/http"
|
|
|
|
"strconv"
|
|
|
|
|
|
|
|
"github.com/gorilla/mux"
|
|
|
|
"github.com/hamburghammer/gsave/db"
|
|
|
|
)
|
|
|
|
|
|
|
|
// NewHostsRouter is a constructor for the HostsRouter.
|
|
|
|
func NewHostsRouter(db db.HostDB) *HostsRouter {
|
|
|
|
return &HostsRouter{db: db}
|
|
|
|
}
|
|
|
|
|
|
|
|
// HostsRouter represents the controller for the hosts routes.
|
|
|
|
type HostsRouter struct {
|
|
|
|
subrouter *mux.Router
|
|
|
|
db db.HostDB
|
|
|
|
}
|
|
|
|
|
|
|
|
// Register registers all routes to the given subrouter.
|
|
|
|
func (hr *HostsRouter) Register(subrouter *mux.Router) {
|
|
|
|
hr.subrouter = subrouter
|
2020-10-20 21:01:53 +02:00
|
|
|
subrouter.HandleFunc("", hr.GetHosts).Methods(http.MethodGet).Name("GetHosts")
|
2020-10-24 15:52:11 +02:00
|
|
|
subrouter.HandleFunc("/{hostname}", hr.GetHost).Methods(http.MethodGet).Name("GetHost")
|
2020-10-26 21:51:54 +01:00
|
|
|
subrouter.HandleFunc("/{hostname}/stats", hr.GetStats).Methods(http.MethodGet).Name("GetStats")
|
2020-10-26 22:29:57 +01:00
|
|
|
subrouter.HandleFunc("/{hostname}/stats", hr.PostStats).Methods(http.MethodPost).Name("PostStats")
|
2020-10-18 15:35:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// GetPrefix returns the the pre route for this controller.
|
|
|
|
func (hr *HostsRouter) GetPrefix() string {
|
|
|
|
return "/hosts"
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetRouteName returns the Name of this controller.
|
|
|
|
func (hr *HostsRouter) GetRouteName() string {
|
|
|
|
return "Hosts"
|
|
|
|
}
|
|
|
|
|
2020-10-24 12:08:46 +02:00
|
|
|
// GetHosts is a HandleFunc to get hosts out of the db with optional pagination as query params.
|
2020-10-20 21:01:53 +02:00
|
|
|
func (hr *HostsRouter) GetHosts(w http.ResponseWriter, r *http.Request) {
|
2020-10-18 15:35:47 +02:00
|
|
|
pagination, err := hr.getSkipAndLimit(r)
|
|
|
|
if err != nil {
|
|
|
|
http.Error(w, err.Error(), http.StatusBadRequest)
|
2020-10-18 18:26:59 +02:00
|
|
|
logBadRequest.Error(err)
|
2020-10-18 15:35:47 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
hosts, err := hr.db.GetHosts(pagination)
|
|
|
|
if err != nil {
|
|
|
|
if errors.Is(err, db.ErrHostsNotFound) || errors.Is(err, db.ErrAllEntriesSkipped) {
|
|
|
|
http.Error(w, err.Error(), http.StatusNotFound)
|
2020-10-18 18:26:59 +02:00
|
|
|
logNotFound.Error(err)
|
2020-10-18 15:35:47 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
http.Error(w, err.Error(), http.StatusInternalServerError)
|
2020-10-18 18:26:59 +02:00
|
|
|
logInternalServerError.Error(err)
|
2020-10-18 15:35:47 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
w.Header().Set("Content-Type", "application/json")
|
|
|
|
json.NewEncoder(w).Encode(hosts)
|
|
|
|
}
|
|
|
|
|
2020-10-24 15:52:11 +02:00
|
|
|
// GetHost is a HandleFunc to get one host. The host name gets read out of the request path.
|
|
|
|
func (hr *HostsRouter) GetHost(w http.ResponseWriter, r *http.Request) {
|
2020-10-18 15:35:47 +02:00
|
|
|
hostname := mux.Vars(r)["hostname"]
|
|
|
|
|
|
|
|
host, err := hr.db.GetHost(hostname)
|
|
|
|
if err != nil {
|
|
|
|
if errors.Is(err, db.ErrHostNotFound) {
|
2020-10-24 15:52:11 +02:00
|
|
|
http.Error(w, fmt.Sprintf("No host with the name '%s' found", hostname), http.StatusNotFound)
|
2020-10-18 18:26:59 +02:00
|
|
|
logNotFound.Error(err)
|
2020-10-18 15:35:47 +02:00
|
|
|
return
|
|
|
|
}
|
2020-10-24 15:52:11 +02:00
|
|
|
http.Error(w, err.Error(), http.StatusInternalServerError)
|
|
|
|
logInternalServerError.Error(err)
|
|
|
|
return
|
2020-10-18 15:35:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
w.Header().Set("Content-Type", "application/json")
|
|
|
|
json.NewEncoder(w).Encode(host)
|
|
|
|
}
|
|
|
|
|
2020-10-26 21:51:54 +01:00
|
|
|
// GetStats is a HandleFunc to get paginated stats for a host.
|
|
|
|
func (hr *HostsRouter) GetStats(w http.ResponseWriter, r *http.Request) {
|
2020-10-18 15:35:47 +02:00
|
|
|
hostname := mux.Vars(r)["hostname"]
|
|
|
|
|
|
|
|
pagination, err := hr.getSkipAndLimit(r)
|
|
|
|
if err != nil {
|
|
|
|
http.Error(w, err.Error(), http.StatusBadRequest)
|
2020-10-18 18:26:59 +02:00
|
|
|
logBadRequest.Error(err)
|
2020-10-18 15:35:47 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
stats, err := hr.db.GetStatsByHostname(hostname, pagination)
|
|
|
|
if err != nil {
|
2020-10-26 21:51:54 +01:00
|
|
|
if errors.Is(err, db.ErrHostNotFound) {
|
|
|
|
http.Error(w, fmt.Sprintf("No host with the name '%s' found", hostname), http.StatusNotFound)
|
|
|
|
logNotFound.Error(err)
|
|
|
|
return
|
|
|
|
} else if errors.Is(err, db.ErrAllEntriesSkipped) {
|
|
|
|
http.Error(w, err.Error(), http.StatusBadRequest)
|
2020-10-18 18:26:59 +02:00
|
|
|
logNotFound.Error(err)
|
2020-10-18 15:35:47 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
http.Error(w, err.Error(), http.StatusInternalServerError)
|
2020-10-18 18:26:59 +02:00
|
|
|
logInternalServerError.Error(err)
|
2020-10-18 15:35:47 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
w.Header().Set("Content-Type", "application/json")
|
|
|
|
json.NewEncoder(w).Encode(stats)
|
|
|
|
}
|
|
|
|
|
2020-10-26 22:29:57 +01:00
|
|
|
// PostStats is a HandleFunc to insert a new data point into the db.
|
|
|
|
func (hr *HostsRouter) PostStats(w http.ResponseWriter, r *http.Request) {
|
2020-10-18 15:35:47 +02:00
|
|
|
hostname := mux.Vars(r)["hostname"]
|
|
|
|
|
|
|
|
var stats db.Stats
|
|
|
|
err := json.NewDecoder(r.Body).Decode(&stats)
|
|
|
|
if err != nil {
|
2020-10-26 22:29:57 +01:00
|
|
|
http.Error(w, "Could not read the body", http.StatusBadRequest)
|
|
|
|
logBadRequest.Error(fmt.Sprintf("JSON error decoding new stat: %v", err))
|
2020-10-18 15:35:47 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
err = hr.db.InsertStats(hostname, stats)
|
|
|
|
if err != nil {
|
|
|
|
http.Error(w, "Something with the DB went wrong.", http.StatusInternalServerError)
|
2020-10-18 18:26:59 +02:00
|
|
|
logInternalServerError.Error(err)
|
2020-10-18 15:35:47 +02:00
|
|
|
return
|
|
|
|
}
|
2020-10-18 17:02:10 +02:00
|
|
|
w.WriteHeader(http.StatusCreated)
|
2020-10-18 15:35:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// getSkipAndLimit from the query of the request.
|
|
|
|
func (hr *HostsRouter) getSkipAndLimit(r *http.Request) (db.Pagination, error) {
|
|
|
|
defaultLimit := "10"
|
|
|
|
defaultSkip := "0"
|
|
|
|
|
|
|
|
strLimit := r.FormValue("limit")
|
|
|
|
if strLimit == "" {
|
|
|
|
strLimit = defaultLimit
|
|
|
|
}
|
|
|
|
limit, err := strconv.ParseInt(strLimit, 10, 64)
|
|
|
|
if err != nil {
|
|
|
|
return db.Pagination{}, fmt.Errorf("Query param 'limit' expected to be a number: %s is not a number", strLimit)
|
|
|
|
}
|
2020-10-24 12:08:46 +02:00
|
|
|
if limit < 0 {
|
|
|
|
return db.Pagination{}, fmt.Errorf("No negative number allowed for the query param 'limit'")
|
|
|
|
}
|
2020-10-18 15:35:47 +02:00
|
|
|
|
|
|
|
strSkip := r.FormValue("skip")
|
|
|
|
if strSkip == "" {
|
|
|
|
strSkip = defaultSkip
|
|
|
|
}
|
|
|
|
skip, err := strconv.ParseInt(strSkip, 10, 64)
|
|
|
|
if err != nil {
|
|
|
|
return db.Pagination{}, fmt.Errorf("Query param 'skip' expected to be a number: %s is not a number", strSkip)
|
|
|
|
}
|
2020-10-24 12:08:46 +02:00
|
|
|
if skip < 0 {
|
|
|
|
return db.Pagination{}, fmt.Errorf("No negative number allowed for the query param 'skip'")
|
|
|
|
}
|
2020-10-18 15:35:47 +02:00
|
|
|
|
|
|
|
return db.Pagination{Skip: int(skip), Limit: int(limit)}, nil
|
|
|
|
}
|