2016-12-25 20:34:02 +00:00
|
|
|
package http
|
|
|
|
|
|
|
|
import (
|
|
|
|
"github.com/portainer/portainer"
|
|
|
|
|
|
|
|
"encoding/json"
|
|
|
|
"log"
|
|
|
|
"net/http"
|
|
|
|
"os"
|
|
|
|
"strconv"
|
|
|
|
|
|
|
|
"github.com/asaskevich/govalidator"
|
|
|
|
"github.com/gorilla/mux"
|
|
|
|
)
|
|
|
|
|
|
|
|
// EndpointHandler represents an HTTP API handler for managing Docker endpoints.
|
|
|
|
type EndpointHandler struct {
|
|
|
|
*mux.Router
|
2017-02-06 05:29:34 +00:00
|
|
|
Logger *log.Logger
|
|
|
|
authorizeEndpointManagement bool
|
|
|
|
EndpointService portainer.EndpointService
|
|
|
|
FileService portainer.FileService
|
2016-12-25 20:34:02 +00:00
|
|
|
}
|
|
|
|
|
2017-02-06 05:29:34 +00:00
|
|
|
const (
|
|
|
|
// ErrEndpointManagementDisabled is an error raised when trying to access the endpoints management endpoints
|
|
|
|
// when the server has been started with the --external-endpoints flag
|
|
|
|
ErrEndpointManagementDisabled = portainer.Error("Endpoint management is disabled")
|
|
|
|
)
|
|
|
|
|
2016-12-25 20:34:02 +00:00
|
|
|
// NewEndpointHandler returns a new instance of EndpointHandler.
|
2017-03-12 16:24:15 +00:00
|
|
|
func NewEndpointHandler(mw *middleWareService) *EndpointHandler {
|
2016-12-25 20:34:02 +00:00
|
|
|
h := &EndpointHandler{
|
2017-03-12 16:24:15 +00:00
|
|
|
Router: mux.NewRouter(),
|
|
|
|
Logger: log.New(os.Stderr, "", log.LstdFlags),
|
|
|
|
}
|
|
|
|
h.Handle("/endpoints",
|
|
|
|
mw.administrator(http.HandlerFunc(h.handlePostEndpoints))).Methods(http.MethodPost)
|
|
|
|
h.Handle("/endpoints",
|
|
|
|
mw.authenticated(http.HandlerFunc(h.handleGetEndpoints))).Methods(http.MethodGet)
|
|
|
|
h.Handle("/endpoints/{id}",
|
|
|
|
mw.administrator(http.HandlerFunc(h.handleGetEndpoint))).Methods(http.MethodGet)
|
|
|
|
h.Handle("/endpoints/{id}",
|
|
|
|
mw.administrator(http.HandlerFunc(h.handlePutEndpoint))).Methods(http.MethodPut)
|
|
|
|
h.Handle("/endpoints/{id}/access",
|
|
|
|
mw.administrator(http.HandlerFunc(h.handlePutEndpointAccess))).Methods(http.MethodPut)
|
|
|
|
h.Handle("/endpoints/{id}",
|
|
|
|
mw.administrator(http.HandlerFunc(h.handleDeleteEndpoint))).Methods(http.MethodDelete)
|
|
|
|
|
2016-12-25 20:34:02 +00:00
|
|
|
return h
|
|
|
|
}
|
|
|
|
|
|
|
|
// handleGetEndpoints handles GET requests on /endpoints
|
|
|
|
func (handler *EndpointHandler) handleGetEndpoints(w http.ResponseWriter, r *http.Request) {
|
|
|
|
endpoints, err := handler.EndpointService.Endpoints()
|
|
|
|
if err != nil {
|
|
|
|
Error(w, err, http.StatusInternalServerError, handler.Logger)
|
|
|
|
return
|
|
|
|
}
|
2017-03-12 16:24:15 +00:00
|
|
|
|
|
|
|
tokenData, err := extractTokenDataFromRequestContext(r)
|
|
|
|
if err != nil {
|
|
|
|
Error(w, err, http.StatusInternalServerError, handler.Logger)
|
|
|
|
}
|
|
|
|
if tokenData == nil {
|
|
|
|
Error(w, portainer.ErrInvalidJWTToken, http.StatusBadRequest, handler.Logger)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
var allowedEndpoints []portainer.Endpoint
|
|
|
|
if tokenData.Role != portainer.AdministratorRole {
|
|
|
|
allowedEndpoints = make([]portainer.Endpoint, 0)
|
|
|
|
for _, endpoint := range endpoints {
|
|
|
|
for _, authorizedUserID := range endpoint.AuthorizedUsers {
|
|
|
|
if authorizedUserID == tokenData.ID {
|
|
|
|
allowedEndpoints = append(allowedEndpoints, endpoint)
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
allowedEndpoints = endpoints
|
|
|
|
}
|
|
|
|
|
|
|
|
encodeJSON(w, allowedEndpoints, handler.Logger)
|
2016-12-25 20:34:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// handlePostEndpoints handles POST requests on /endpoints
|
|
|
|
func (handler *EndpointHandler) handlePostEndpoints(w http.ResponseWriter, r *http.Request) {
|
2017-02-06 05:29:34 +00:00
|
|
|
if !handler.authorizeEndpointManagement {
|
|
|
|
Error(w, ErrEndpointManagementDisabled, http.StatusServiceUnavailable, handler.Logger)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2016-12-25 20:34:02 +00:00
|
|
|
var req postEndpointsRequest
|
|
|
|
if err := json.NewDecoder(r.Body).Decode(&req); err != nil {
|
|
|
|
Error(w, ErrInvalidJSON, http.StatusBadRequest, handler.Logger)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
_, err := govalidator.ValidateStruct(req)
|
|
|
|
if err != nil {
|
|
|
|
Error(w, ErrInvalidRequestFormat, http.StatusBadRequest, handler.Logger)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
endpoint := &portainer.Endpoint{
|
2017-03-12 16:24:15 +00:00
|
|
|
Name: req.Name,
|
|
|
|
URL: req.URL,
|
|
|
|
TLS: req.TLS,
|
|
|
|
AuthorizedUsers: []portainer.UserID{},
|
2016-12-25 20:34:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
err = handler.EndpointService.CreateEndpoint(endpoint)
|
|
|
|
if err != nil {
|
|
|
|
Error(w, err, http.StatusInternalServerError, handler.Logger)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if req.TLS {
|
|
|
|
caCertPath, _ := handler.FileService.GetPathForTLSFile(endpoint.ID, portainer.TLSFileCA)
|
|
|
|
endpoint.TLSCACertPath = caCertPath
|
|
|
|
certPath, _ := handler.FileService.GetPathForTLSFile(endpoint.ID, portainer.TLSFileCert)
|
|
|
|
endpoint.TLSCertPath = certPath
|
|
|
|
keyPath, _ := handler.FileService.GetPathForTLSFile(endpoint.ID, portainer.TLSFileKey)
|
|
|
|
endpoint.TLSKeyPath = keyPath
|
|
|
|
err = handler.EndpointService.UpdateEndpoint(endpoint.ID, endpoint)
|
|
|
|
if err != nil {
|
|
|
|
Error(w, err, http.StatusInternalServerError, handler.Logger)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
encodeJSON(w, &postEndpointsResponse{ID: int(endpoint.ID)}, handler.Logger)
|
|
|
|
}
|
|
|
|
|
|
|
|
type postEndpointsRequest struct {
|
|
|
|
Name string `valid:"required"`
|
|
|
|
URL string `valid:"required"`
|
|
|
|
TLS bool
|
|
|
|
}
|
|
|
|
|
|
|
|
type postEndpointsResponse struct {
|
|
|
|
ID int `json:"Id"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// handleGetEndpoint handles GET requests on /endpoints/:id
|
|
|
|
func (handler *EndpointHandler) handleGetEndpoint(w http.ResponseWriter, r *http.Request) {
|
|
|
|
vars := mux.Vars(r)
|
|
|
|
id := vars["id"]
|
|
|
|
|
|
|
|
endpointID, err := strconv.Atoi(id)
|
|
|
|
if err != nil {
|
|
|
|
Error(w, err, http.StatusBadRequest, handler.Logger)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2017-03-12 16:24:15 +00:00
|
|
|
endpoint, err := handler.EndpointService.Endpoint(portainer.EndpointID(endpointID))
|
|
|
|
if err == portainer.ErrEndpointNotFound {
|
|
|
|
Error(w, err, http.StatusNotFound, handler.Logger)
|
|
|
|
return
|
|
|
|
} else if err != nil {
|
|
|
|
Error(w, err, http.StatusInternalServerError, handler.Logger)
|
|
|
|
return
|
2016-12-25 20:34:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
encodeJSON(w, endpoint, handler.Logger)
|
|
|
|
}
|
|
|
|
|
2017-03-12 16:24:15 +00:00
|
|
|
// handlePutEndpointAccess handles PUT requests on /endpoints/:id/access
|
|
|
|
func (handler *EndpointHandler) handlePutEndpointAccess(w http.ResponseWriter, r *http.Request) {
|
2016-12-25 20:34:02 +00:00
|
|
|
vars := mux.Vars(r)
|
|
|
|
id := vars["id"]
|
|
|
|
|
|
|
|
endpointID, err := strconv.Atoi(id)
|
|
|
|
if err != nil {
|
|
|
|
Error(w, err, http.StatusBadRequest, handler.Logger)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2017-03-12 16:24:15 +00:00
|
|
|
var req putEndpointAccessRequest
|
|
|
|
if err = json.NewDecoder(r.Body).Decode(&req); err != nil {
|
|
|
|
Error(w, ErrInvalidJSON, http.StatusBadRequest, handler.Logger)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
_, err = govalidator.ValidateStruct(req)
|
|
|
|
if err != nil {
|
|
|
|
Error(w, ErrInvalidRequestFormat, http.StatusBadRequest, handler.Logger)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2016-12-25 20:34:02 +00:00
|
|
|
endpoint, err := handler.EndpointService.Endpoint(portainer.EndpointID(endpointID))
|
|
|
|
if err == portainer.ErrEndpointNotFound {
|
|
|
|
Error(w, err, http.StatusNotFound, handler.Logger)
|
|
|
|
return
|
|
|
|
} else if err != nil {
|
|
|
|
Error(w, err, http.StatusInternalServerError, handler.Logger)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2017-03-12 16:24:15 +00:00
|
|
|
authorizedUserIDs := []portainer.UserID{}
|
|
|
|
for _, value := range req.AuthorizedUsers {
|
|
|
|
authorizedUserIDs = append(authorizedUserIDs, portainer.UserID(value))
|
|
|
|
}
|
|
|
|
endpoint.AuthorizedUsers = authorizedUserIDs
|
|
|
|
|
|
|
|
err = handler.EndpointService.UpdateEndpoint(endpoint.ID, endpoint)
|
2016-12-25 20:34:02 +00:00
|
|
|
if err != nil {
|
|
|
|
Error(w, err, http.StatusInternalServerError, handler.Logger)
|
2017-03-12 16:24:15 +00:00
|
|
|
return
|
2016-12-25 20:34:02 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-03-12 16:24:15 +00:00
|
|
|
type putEndpointAccessRequest struct {
|
2017-03-17 10:52:17 +00:00
|
|
|
AuthorizedUsers []int `valid:"-"`
|
2017-03-12 16:24:15 +00:00
|
|
|
}
|
|
|
|
|
2016-12-25 20:34:02 +00:00
|
|
|
// handlePutEndpoint handles PUT requests on /endpoints/:id
|
|
|
|
func (handler *EndpointHandler) handlePutEndpoint(w http.ResponseWriter, r *http.Request) {
|
2017-02-06 05:29:34 +00:00
|
|
|
if !handler.authorizeEndpointManagement {
|
|
|
|
Error(w, ErrEndpointManagementDisabled, http.StatusServiceUnavailable, handler.Logger)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2016-12-25 20:34:02 +00:00
|
|
|
vars := mux.Vars(r)
|
|
|
|
id := vars["id"]
|
|
|
|
|
|
|
|
endpointID, err := strconv.Atoi(id)
|
|
|
|
if err != nil {
|
|
|
|
Error(w, err, http.StatusBadRequest, handler.Logger)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
var req putEndpointsRequest
|
|
|
|
if err = json.NewDecoder(r.Body).Decode(&req); err != nil {
|
|
|
|
Error(w, ErrInvalidJSON, http.StatusBadRequest, handler.Logger)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
_, err = govalidator.ValidateStruct(req)
|
|
|
|
if err != nil {
|
|
|
|
Error(w, ErrInvalidRequestFormat, http.StatusBadRequest, handler.Logger)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2017-03-12 16:24:15 +00:00
|
|
|
endpoint, err := handler.EndpointService.Endpoint(portainer.EndpointID(endpointID))
|
|
|
|
if err == portainer.ErrEndpointNotFound {
|
|
|
|
Error(w, err, http.StatusNotFound, handler.Logger)
|
|
|
|
return
|
|
|
|
} else if err != nil {
|
|
|
|
Error(w, err, http.StatusInternalServerError, handler.Logger)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if req.Name != "" {
|
|
|
|
endpoint.Name = req.Name
|
|
|
|
}
|
|
|
|
|
|
|
|
if req.URL != "" {
|
|
|
|
endpoint.URL = req.URL
|
2016-12-25 20:34:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if req.TLS {
|
2017-03-12 16:24:15 +00:00
|
|
|
endpoint.TLS = true
|
2016-12-25 20:34:02 +00:00
|
|
|
caCertPath, _ := handler.FileService.GetPathForTLSFile(endpoint.ID, portainer.TLSFileCA)
|
|
|
|
endpoint.TLSCACertPath = caCertPath
|
|
|
|
certPath, _ := handler.FileService.GetPathForTLSFile(endpoint.ID, portainer.TLSFileCert)
|
|
|
|
endpoint.TLSCertPath = certPath
|
|
|
|
keyPath, _ := handler.FileService.GetPathForTLSFile(endpoint.ID, portainer.TLSFileKey)
|
|
|
|
endpoint.TLSKeyPath = keyPath
|
|
|
|
} else {
|
2017-03-12 16:24:15 +00:00
|
|
|
endpoint.TLS = false
|
|
|
|
endpoint.TLSCACertPath = ""
|
|
|
|
endpoint.TLSCertPath = ""
|
|
|
|
endpoint.TLSKeyPath = ""
|
2016-12-25 20:34:02 +00:00
|
|
|
err = handler.FileService.DeleteTLSFiles(endpoint.ID)
|
|
|
|
if err != nil {
|
|
|
|
Error(w, err, http.StatusInternalServerError, handler.Logger)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
err = handler.EndpointService.UpdateEndpoint(endpoint.ID, endpoint)
|
|
|
|
if err != nil {
|
|
|
|
Error(w, err, http.StatusInternalServerError, handler.Logger)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
type putEndpointsRequest struct {
|
2017-03-12 16:24:15 +00:00
|
|
|
Name string `valid:"-"`
|
|
|
|
URL string `valid:"-"`
|
|
|
|
TLS bool `valid:"-"`
|
2016-12-25 20:34:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// handleDeleteEndpoint handles DELETE requests on /endpoints/:id
|
|
|
|
func (handler *EndpointHandler) handleDeleteEndpoint(w http.ResponseWriter, r *http.Request) {
|
2017-02-06 05:29:34 +00:00
|
|
|
if !handler.authorizeEndpointManagement {
|
|
|
|
Error(w, ErrEndpointManagementDisabled, http.StatusServiceUnavailable, handler.Logger)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2016-12-25 20:34:02 +00:00
|
|
|
vars := mux.Vars(r)
|
|
|
|
id := vars["id"]
|
|
|
|
|
|
|
|
endpointID, err := strconv.Atoi(id)
|
|
|
|
if err != nil {
|
|
|
|
Error(w, err, http.StatusBadRequest, handler.Logger)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2017-03-12 16:24:15 +00:00
|
|
|
endpoint, err := handler.EndpointService.Endpoint(portainer.EndpointID(endpointID))
|
2017-01-22 23:14:34 +00:00
|
|
|
|
2016-12-25 20:34:02 +00:00
|
|
|
if err == portainer.ErrEndpointNotFound {
|
|
|
|
Error(w, err, http.StatusNotFound, handler.Logger)
|
|
|
|
return
|
|
|
|
} else if err != nil {
|
|
|
|
Error(w, err, http.StatusInternalServerError, handler.Logger)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
err = handler.EndpointService.DeleteEndpoint(portainer.EndpointID(endpointID))
|
|
|
|
if err != nil {
|
|
|
|
Error(w, err, http.StatusInternalServerError, handler.Logger)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if endpoint.TLS {
|
|
|
|
err = handler.FileService.DeleteTLSFiles(portainer.EndpointID(endpointID))
|
|
|
|
if err != nil {
|
|
|
|
Error(w, err, http.StatusInternalServerError, handler.Logger)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|