mirror of https://github.com/portainer/portainer
71 lines
2.7 KiB
Go
71 lines
2.7 KiB
Go
package docker
|
|
|
|
import (
|
|
"errors"
|
|
"net/http"
|
|
|
|
"github.com/portainer/portainer/api/dataservices"
|
|
"github.com/portainer/portainer/api/docker"
|
|
dockerclient "github.com/portainer/portainer/api/docker/client"
|
|
"github.com/portainer/portainer/api/http/handler/docker/containers"
|
|
"github.com/portainer/portainer/api/http/handler/docker/images"
|
|
"github.com/portainer/portainer/api/http/middlewares"
|
|
"github.com/portainer/portainer/api/http/security"
|
|
"github.com/portainer/portainer/api/internal/authorization"
|
|
"github.com/portainer/portainer/api/internal/endpointutils"
|
|
httperror "github.com/portainer/portainer/pkg/libhttp/error"
|
|
|
|
"github.com/gorilla/mux"
|
|
)
|
|
|
|
// Handler is the HTTP handler which will natively deal with to external environments(endpoints).
|
|
type Handler struct {
|
|
*mux.Router
|
|
requestBouncer security.BouncerService
|
|
dataStore dataservices.DataStore
|
|
dockerClientFactory *dockerclient.ClientFactory
|
|
authorizationService *authorization.Service
|
|
containerService *docker.ContainerService
|
|
}
|
|
|
|
// NewHandler creates a handler to process non-proxied requests to docker APIs directly.
|
|
func NewHandler(bouncer security.BouncerService, authorizationService *authorization.Service, dataStore dataservices.DataStore, dockerClientFactory *dockerclient.ClientFactory, containerService *docker.ContainerService) *Handler {
|
|
h := &Handler{
|
|
Router: mux.NewRouter(),
|
|
requestBouncer: bouncer,
|
|
authorizationService: authorizationService,
|
|
dataStore: dataStore,
|
|
dockerClientFactory: dockerClientFactory,
|
|
containerService: containerService,
|
|
}
|
|
|
|
// endpoints
|
|
endpointRouter := h.PathPrefix("/{id}").Subrouter()
|
|
endpointRouter.Use(middlewares.WithEndpoint(dataStore.Endpoint(), "id"))
|
|
endpointRouter.Use(dockerOnlyMiddleware)
|
|
|
|
containersHandler := containers.NewHandler("/{id}/containers", bouncer, dataStore, dockerClientFactory, containerService)
|
|
endpointRouter.PathPrefix("/containers").Handler(containersHandler)
|
|
|
|
imagesHandler := images.NewHandler("/{id}/images", bouncer, dockerClientFactory)
|
|
endpointRouter.PathPrefix("/images").Handler(imagesHandler)
|
|
return h
|
|
}
|
|
|
|
func dockerOnlyMiddleware(next http.Handler) http.Handler {
|
|
return http.HandlerFunc(func(rw http.ResponseWriter, request *http.Request) {
|
|
endpoint, err := middlewares.FetchEndpoint(request)
|
|
if err != nil {
|
|
httperror.WriteError(rw, http.StatusInternalServerError, "Unable to find an environment on request context", err)
|
|
return
|
|
}
|
|
|
|
if !endpointutils.IsDockerEndpoint(endpoint) {
|
|
errMessage := "environment is not a docker environment"
|
|
httperror.WriteError(rw, http.StatusBadRequest, errMessage, errors.New(errMessage))
|
|
return
|
|
}
|
|
next.ServeHTTP(rw, request)
|
|
})
|
|
}
|