2019-04-09 17:34:40 +00:00
|
|
|
package router
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"encoding/json"
|
|
|
|
"net/http"
|
|
|
|
"strconv"
|
|
|
|
|
|
|
|
"github.com/gorilla/mux"
|
|
|
|
"github.com/pkg/errors"
|
|
|
|
|
2020-08-05 09:11:24 +00:00
|
|
|
"github.com/rootless-containers/rootlesskit/pkg/api"
|
2019-04-09 17:34:40 +00:00
|
|
|
"github.com/rootless-containers/rootlesskit/pkg/port"
|
2021-03-02 08:13:01 +00:00
|
|
|
"github.com/rootless-containers/rootlesskit/pkg/version"
|
2019-04-09 17:34:40 +00:00
|
|
|
)
|
|
|
|
|
2021-03-02 08:13:01 +00:00
|
|
|
// NetworkDriver is implemented by network.ParentDriver
|
|
|
|
type NetworkDriver interface {
|
|
|
|
Info(context.Context) (*api.NetworkDriverInfo, error)
|
|
|
|
}
|
|
|
|
|
|
|
|
// PortDriver is implemented by port.ParentDriver
|
|
|
|
type PortDriver interface {
|
|
|
|
Info(context.Context) (*api.PortDriverInfo, error)
|
|
|
|
port.Manager
|
|
|
|
}
|
|
|
|
|
2019-04-09 17:34:40 +00:00
|
|
|
type Backend struct {
|
2021-03-02 08:13:01 +00:00
|
|
|
StateDir string
|
|
|
|
ChildPID int
|
|
|
|
// NetworkDriver can be nil
|
|
|
|
NetworkDriver NetworkDriver
|
2019-04-09 17:34:40 +00:00
|
|
|
// PortDriver MUST be thread-safe.
|
|
|
|
// PortDriver can be nil
|
2021-03-02 08:13:01 +00:00
|
|
|
PortDriver PortDriver
|
2019-04-09 17:34:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
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
|
2020-08-05 09:11:24 +00:00
|
|
|
e := api.ErrorJSON{
|
2019-04-09 17:34:40 +00:00
|
|
|
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)
|
|
|
|
}
|
|
|
|
|
2021-03-02 08:13:01 +00:00
|
|
|
func (b *Backend) GetInfo(w http.ResponseWriter, r *http.Request) {
|
|
|
|
info := &api.Info{
|
|
|
|
APIVersion: api.Version,
|
|
|
|
Version: version.Version,
|
|
|
|
StateDir: b.StateDir,
|
|
|
|
ChildPID: b.ChildPID,
|
|
|
|
}
|
|
|
|
if b.NetworkDriver != nil {
|
|
|
|
ndInfo, err := b.NetworkDriver.Info(context.Background())
|
|
|
|
if err != nil {
|
|
|
|
b.onError(w, r, err, http.StatusInternalServerError)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
info.NetworkDriver = ndInfo
|
|
|
|
}
|
|
|
|
if b.PortDriver != nil {
|
|
|
|
pdInfo, err := b.PortDriver.Info(context.Background())
|
|
|
|
if err != nil {
|
|
|
|
b.onError(w, r, err, http.StatusInternalServerError)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
info.PortDriver = pdInfo
|
|
|
|
}
|
|
|
|
m, err := json.Marshal(info)
|
|
|
|
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)
|
|
|
|
}
|
|
|
|
|
2019-04-09 17:34:40 +00:00
|
|
|
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)
|
2021-03-02 08:13:01 +00:00
|
|
|
v1.Path("/info").Methods("GET").HandlerFunc(b.GetInfo)
|
2019-04-09 17:34:40 +00:00
|
|
|
}
|