2015-09-15 19:01:49 -04:00
|
|
|
package server
|
|
|
|
|
|
|
|
import (
|
|
|
|
"net/http"
|
|
|
|
"runtime"
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
"github.com/Sirupsen/logrus"
|
|
|
|
"github.com/docker/docker/api"
|
2015-09-23 19:42:08 -04:00
|
|
|
"github.com/docker/docker/api/server/httputils"
|
2015-09-15 19:01:49 -04:00
|
|
|
"github.com/docker/docker/autogen/dockerversion"
|
|
|
|
"github.com/docker/docker/errors"
|
|
|
|
"github.com/docker/docker/pkg/version"
|
2015-09-29 17:32:07 -04:00
|
|
|
"golang.org/x/net/context"
|
2015-09-15 19:01:49 -04:00
|
|
|
)
|
|
|
|
|
|
|
|
// middleware is an adapter to allow the use of ordinary functions as Docker API filters.
|
|
|
|
// Any function that has the appropriate signature can be register as a middleware.
|
2015-09-23 19:42:08 -04:00
|
|
|
type middleware func(handler httputils.APIFunc) httputils.APIFunc
|
2015-09-15 19:01:49 -04:00
|
|
|
|
|
|
|
// loggingMiddleware logs each request when logging is enabled.
|
2015-09-23 19:42:08 -04:00
|
|
|
func (s *Server) loggingMiddleware(handler httputils.APIFunc) httputils.APIFunc {
|
2015-09-15 19:01:49 -04:00
|
|
|
return func(ctx context.Context, w http.ResponseWriter, r *http.Request, vars map[string]string) error {
|
|
|
|
if s.cfg.Logging {
|
|
|
|
logrus.Infof("%s %s", r.Method, r.RequestURI)
|
|
|
|
}
|
|
|
|
return handler(ctx, w, r, vars)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// userAgentMiddleware checks the User-Agent header looking for a valid docker client spec.
|
2015-09-23 19:42:08 -04:00
|
|
|
func (s *Server) userAgentMiddleware(handler httputils.APIFunc) httputils.APIFunc {
|
2015-09-15 19:01:49 -04:00
|
|
|
return func(ctx context.Context, w http.ResponseWriter, r *http.Request, vars map[string]string) error {
|
|
|
|
if strings.Contains(r.Header.Get("User-Agent"), "Docker-Client/") {
|
|
|
|
dockerVersion := version.Version(s.cfg.Version)
|
|
|
|
|
|
|
|
userAgent := strings.Split(r.Header.Get("User-Agent"), "/")
|
|
|
|
|
|
|
|
// v1.20 onwards includes the GOOS of the client after the version
|
|
|
|
// such as Docker/1.7.0 (linux)
|
|
|
|
if len(userAgent) == 2 && strings.Contains(userAgent[1], " ") {
|
|
|
|
userAgent[1] = strings.Split(userAgent[1], " ")[0]
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(userAgent) == 2 && !dockerVersion.Equal(version.Version(userAgent[1])) {
|
|
|
|
logrus.Debugf("Warning: client and server don't have the same version (client: %s, server: %s)", userAgent[1], dockerVersion)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return handler(ctx, w, r, vars)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// corsMiddleware sets the CORS header expectations in the server.
|
2015-09-23 19:42:08 -04:00
|
|
|
func (s *Server) corsMiddleware(handler httputils.APIFunc) httputils.APIFunc {
|
2015-09-15 19:01:49 -04:00
|
|
|
return func(ctx context.Context, w http.ResponseWriter, r *http.Request, vars map[string]string) error {
|
|
|
|
// If "api-cors-header" is not given, but "api-enable-cors" is true, we set cors to "*"
|
|
|
|
// otherwise, all head values will be passed to HTTP handler
|
|
|
|
corsHeaders := s.cfg.CorsHeaders
|
|
|
|
if corsHeaders == "" && s.cfg.EnableCors {
|
|
|
|
corsHeaders = "*"
|
|
|
|
}
|
|
|
|
|
|
|
|
if corsHeaders != "" {
|
|
|
|
writeCorsHeaders(w, r, corsHeaders)
|
|
|
|
}
|
|
|
|
return handler(ctx, w, r, vars)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// versionMiddleware checks the api version requirements before passing the request to the server handler.
|
2015-09-23 19:42:08 -04:00
|
|
|
func versionMiddleware(handler httputils.APIFunc) httputils.APIFunc {
|
2015-09-15 19:01:49 -04:00
|
|
|
return func(ctx context.Context, w http.ResponseWriter, r *http.Request, vars map[string]string) error {
|
|
|
|
apiVersion := version.Version(vars["version"])
|
|
|
|
if apiVersion == "" {
|
|
|
|
apiVersion = api.Version
|
|
|
|
}
|
|
|
|
|
|
|
|
if apiVersion.GreaterThan(api.Version) {
|
|
|
|
return errors.ErrorCodeNewerClientVersion.WithArgs(apiVersion, api.Version)
|
|
|
|
}
|
|
|
|
if apiVersion.LessThan(api.MinVersion) {
|
|
|
|
return errors.ErrorCodeOldClientVersion.WithArgs(apiVersion, api.Version)
|
|
|
|
}
|
|
|
|
|
|
|
|
w.Header().Set("Server", "Docker/"+dockerversion.VERSION+" ("+runtime.GOOS+")")
|
2015-09-23 19:42:08 -04:00
|
|
|
ctx = context.WithValue(ctx, httputils.APIVersionKey, apiVersion)
|
2015-09-15 19:01:49 -04:00
|
|
|
return handler(ctx, w, r, vars)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// handleWithGlobalMiddlwares wraps the handler function for a request with
|
|
|
|
// the server's global middlewares. The order of the middlewares is backwards,
|
|
|
|
// meaning that the first in the list will be evaludated last.
|
|
|
|
//
|
|
|
|
// Example: handleWithGlobalMiddlewares(s.getContainersName)
|
|
|
|
//
|
|
|
|
// s.loggingMiddleware(
|
|
|
|
// s.userAgentMiddleware(
|
|
|
|
// s.corsMiddleware(
|
|
|
|
// versionMiddleware(s.getContainersName)
|
|
|
|
// )
|
|
|
|
// )
|
|
|
|
// )
|
2015-09-23 19:42:08 -04:00
|
|
|
// )
|
|
|
|
func (s *Server) handleWithGlobalMiddlewares(handler httputils.APIFunc) httputils.APIFunc {
|
2015-09-15 19:01:49 -04:00
|
|
|
middlewares := []middleware{
|
|
|
|
versionMiddleware,
|
|
|
|
s.corsMiddleware,
|
|
|
|
s.userAgentMiddleware,
|
|
|
|
s.loggingMiddleware,
|
|
|
|
}
|
|
|
|
|
|
|
|
h := handler
|
|
|
|
for _, m := range middlewares {
|
|
|
|
h = m(h)
|
|
|
|
}
|
|
|
|
return h
|
|
|
|
}
|