1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586 |
- package middleware // import "github.com/docker/docker/api/server/middleware"
- import (
- "context"
- "fmt"
- "net/http"
- "runtime"
- "github.com/docker/docker/api"
- "github.com/docker/docker/api/server/httputils"
- "github.com/docker/docker/api/types/versions"
- )
- // VersionMiddleware is a middleware that
- // validates the client and server versions.
- type VersionMiddleware struct {
- serverVersion string
- // defaultAPIVersion is the default API version provided by the API server,
- // specified as "major.minor". It is usually configured to the latest API
- // version [github.com/docker/docker/api.DefaultVersion].
- //
- // API requests for API versions greater than this version are rejected by
- // the server and produce a [versionUnsupportedError].
- defaultAPIVersion string
- // minAPIVersion is the minimum API version provided by the API server,
- // specified as "major.minor".
- //
- // API requests for API versions lower than this version are rejected by
- // the server and produce a [versionUnsupportedError].
- minAPIVersion string
- }
- // NewVersionMiddleware creates a VersionMiddleware with the given versions.
- func NewVersionMiddleware(serverVersion, defaultAPIVersion, minAPIVersion string) (*VersionMiddleware, error) {
- if versions.LessThan(defaultAPIVersion, api.MinSupportedAPIVersion) || versions.GreaterThan(defaultAPIVersion, api.DefaultVersion) {
- return nil, fmt.Errorf("invalid default API version (%s): must be between %s and %s", defaultAPIVersion, api.MinSupportedAPIVersion, api.DefaultVersion)
- }
- if versions.LessThan(minAPIVersion, api.MinSupportedAPIVersion) || versions.GreaterThan(minAPIVersion, api.DefaultVersion) {
- return nil, fmt.Errorf("invalid minimum API version (%s): must be between %s and %s", minAPIVersion, api.MinSupportedAPIVersion, api.DefaultVersion)
- }
- if versions.GreaterThan(minAPIVersion, defaultAPIVersion) {
- return nil, fmt.Errorf("invalid API version: the minimum API version (%s) is higher than the default version (%s)", minAPIVersion, defaultAPIVersion)
- }
- return &VersionMiddleware{
- serverVersion: serverVersion,
- defaultAPIVersion: defaultAPIVersion,
- minAPIVersion: minAPIVersion,
- }, nil
- }
- type versionUnsupportedError struct {
- version, minVersion, maxVersion string
- }
- func (e versionUnsupportedError) Error() string {
- if e.minVersion != "" {
- return fmt.Sprintf("client version %s is too old. Minimum supported API version is %s, please upgrade your client to a newer version", e.version, e.minVersion)
- }
- return fmt.Sprintf("client version %s is too new. Maximum supported API version is %s", e.version, e.maxVersion)
- }
- func (e versionUnsupportedError) InvalidParameter() {}
- // WrapHandler returns a new handler function wrapping the previous one in the request chain.
- func (v VersionMiddleware) WrapHandler(handler func(ctx context.Context, w http.ResponseWriter, r *http.Request, vars map[string]string) error) func(ctx context.Context, w http.ResponseWriter, r *http.Request, vars map[string]string) error {
- return func(ctx context.Context, w http.ResponseWriter, r *http.Request, vars map[string]string) error {
- w.Header().Set("Server", fmt.Sprintf("Docker/%s (%s)", v.serverVersion, runtime.GOOS))
- w.Header().Set("API-Version", v.defaultAPIVersion)
- w.Header().Set("OSType", runtime.GOOS)
- apiVersion := vars["version"]
- if apiVersion == "" {
- apiVersion = v.defaultAPIVersion
- }
- if versions.LessThan(apiVersion, v.minAPIVersion) {
- return versionUnsupportedError{version: apiVersion, minVersion: v.minAPIVersion}
- }
- if versions.GreaterThan(apiVersion, v.defaultAPIVersion) {
- return versionUnsupportedError{version: apiVersion, maxVersion: v.defaultAPIVersion}
- }
- ctx = context.WithValue(ctx, httputils.APIVersionKey{}, apiVersion)
- return handler(ctx, w, r, vars)
- }
- }
|