k3s/vendor/github.com/rootless-containers/rootlesskit/pkg/api/router/router.go

115 lines
2.9 KiB
Go

package router
import (
"context"
"encoding/json"
"net/http"
"strconv"
"github.com/gorilla/mux"
"github.com/pkg/errors"
"github.com/rootless-containers/rootlesskit/pkg/port"
)
type Backend struct {
// PortDriver MUST be thread-safe.
// PortDriver can be nil
PortDriver port.ParentDriver
}
func (b *Backend) onError(w http.ResponseWriter, r *http.Request, err error, ec int) {
w.WriteHeader(ec)
w.Header().Set("Content-Type", "application/json")
// it is safe to return the err to the client, because the client is reliable
type errorJSON struct {
Message string `json:"message"`
}
e := errorJSON{
Message: err.Error(),
}
_ = json.NewEncoder(w).Encode(e)
}
func (b *Backend) onPortDriverNil(w http.ResponseWriter, r *http.Request) {
b.onError(w, r, errors.New("no PortDriver is available"), http.StatusBadRequest)
}
// GetPorts is handler for GET /v{N}/ports
func (b *Backend) GetPorts(w http.ResponseWriter, r *http.Request) {
if b.PortDriver == nil {
b.onPortDriverNil(w, r)
return
}
ports, err := b.PortDriver.ListPorts(context.TODO())
if err != nil {
b.onError(w, r, err, http.StatusInternalServerError)
return
}
m, err := json.Marshal(ports)
if err != nil {
b.onError(w, r, err, http.StatusInternalServerError)
return
}
w.Header().Set("Content-Type", "application/json")
w.WriteHeader(http.StatusOK)
w.Write(m)
}
// PostPort is the handler for POST /v{N}/ports
func (b *Backend) PostPort(w http.ResponseWriter, r *http.Request) {
if b.PortDriver == nil {
b.onPortDriverNil(w, r)
return
}
decoder := json.NewDecoder(r.Body)
var portSpec port.Spec
if err := decoder.Decode(&portSpec); err != nil {
b.onError(w, r, err, http.StatusBadRequest)
return
}
portStatus, err := b.PortDriver.AddPort(context.TODO(), portSpec)
if err != nil {
b.onError(w, r, err, http.StatusBadRequest)
return
}
m, err := json.Marshal(portStatus)
if err != nil {
b.onError(w, r, err, http.StatusInternalServerError)
return
}
w.Header().Set("Content-Type", "application/json")
w.WriteHeader(http.StatusCreated)
w.Write(m)
}
// DeletePort is the handler for POST /v{N}/ports/{id}
func (b *Backend) DeletePort(w http.ResponseWriter, r *http.Request) {
if b.PortDriver == nil {
b.onPortDriverNil(w, r)
return
}
idStr, ok := mux.Vars(r)["id"]
if !ok {
b.onError(w, r, errors.New("id not specified"), http.StatusBadRequest)
return
}
id, err := strconv.Atoi(idStr)
if err != nil {
b.onError(w, r, errors.Wrapf(err, "bad id %s", idStr), http.StatusBadRequest)
return
}
if err := b.PortDriver.RemovePort(context.TODO(), id); err != nil {
b.onError(w, r, err, http.StatusBadRequest)
return
}
w.WriteHeader(http.StatusOK)
}
func AddRoutes(r *mux.Router, b *Backend) {
v1 := r.PathPrefix("/v1").Subrouter()
v1.Path("/ports").Methods("GET").HandlerFunc(b.GetPorts)
v1.Path("/ports").Methods("POST").HandlerFunc(b.PostPort)
v1.Path("/ports/{id}").Methods("DELETE").HandlerFunc(b.DeletePort)
}