mirror of
				https://github.com/k3s-io/kubernetes.git
				synced 2025-10-24 17:10:44 +00:00 
			
		
		
		
	**What this PR does / why we need it**: The --pod-infra-container-image parameter allows the user to specify an arbitrary image to be used as the pod infra container (AKA sandbox), an internal piece of the dockershim implementation of the Container Runtime Interface. The dockershim does not have access to any of the pod-level image pull credentials configuration, so if the user specifies an image from a private registry, the image pull will fail. This change allows the dockershim to read local docker configuration (e.g. /root/.docker/config.json) and use it when pulling the pod infra container image. **Which issue this PR fixes**: fixes #45738 **Special notes for your reviewer**: The changes to fake_client for writing local config files deserve some attention. **Release note**: ```release-note NONE ```
		
			
				
	
	
		
			454 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			454 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| /*
 | |
| Copyright 2016 The Kubernetes Authors.
 | |
| 
 | |
| Licensed under the Apache License, Version 2.0 (the "License");
 | |
| you may not use this file except in compliance with the License.
 | |
| You may obtain a copy of the License at
 | |
| 
 | |
|     http://www.apache.org/licenses/LICENSE-2.0
 | |
| 
 | |
| Unless required by applicable law or agreed to in writing, software
 | |
| distributed under the License is distributed on an "AS IS" BASIS,
 | |
| WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | |
| See the License for the specific language governing permissions and
 | |
| limitations under the License.
 | |
| */
 | |
| 
 | |
| package dockershim
 | |
| 
 | |
| import (
 | |
| 	"bytes"
 | |
| 	"crypto/md5"
 | |
| 	"encoding/json"
 | |
| 	"fmt"
 | |
| 	"io/ioutil"
 | |
| 	"os"
 | |
| 	"path/filepath"
 | |
| 	"regexp"
 | |
| 	"strconv"
 | |
| 	"strings"
 | |
| 
 | |
| 	"github.com/blang/semver"
 | |
| 	dockertypes "github.com/docker/engine-api/types"
 | |
| 	dockerfilters "github.com/docker/engine-api/types/filters"
 | |
| 	dockernat "github.com/docker/go-connections/nat"
 | |
| 	"github.com/golang/glog"
 | |
| 
 | |
| 	"k8s.io/kubernetes/pkg/api/v1"
 | |
| 	"k8s.io/kubernetes/pkg/credentialprovider"
 | |
| 	runtimeapi "k8s.io/kubernetes/pkg/kubelet/apis/cri/v1alpha1"
 | |
| 	"k8s.io/kubernetes/pkg/kubelet/types"
 | |
| 	"k8s.io/kubernetes/pkg/security/apparmor"
 | |
| 
 | |
| 	"k8s.io/kubernetes/pkg/kubelet/dockershim/libdocker"
 | |
| )
 | |
| 
 | |
| const (
 | |
| 	annotationPrefix = "annotation."
 | |
| 
 | |
| 	// Docker changed the API for specifying options in v1.11
 | |
| 	securityOptSeparatorChangeVersion = "1.23.0" // Corresponds to docker 1.11.x
 | |
| 	securityOptSeparatorOld           = ':'
 | |
| 	securityOptSeparatorNew           = '='
 | |
| )
 | |
| 
 | |
| var (
 | |
| 	conflictRE = regexp.MustCompile(`Conflict. (?:.)+ is already in use by container ([0-9a-z]+)`)
 | |
| 
 | |
| 	// Docker changes the security option separator from ':' to '=' in the 1.23
 | |
| 	// API version.
 | |
| 	optsSeparatorChangeVersion = semver.MustParse(securityOptSeparatorChangeVersion)
 | |
| 
 | |
| 	defaultSeccompOpt = []dockerOpt{{"seccomp", "unconfined", ""}}
 | |
| )
 | |
| 
 | |
| // generateEnvList converts KeyValue list to a list of strings, in the form of
 | |
| // '<key>=<value>', which can be understood by docker.
 | |
| func generateEnvList(envs []*runtimeapi.KeyValue) (result []string) {
 | |
| 	for _, env := range envs {
 | |
| 		result = append(result, fmt.Sprintf("%s=%s", env.Key, env.Value))
 | |
| 	}
 | |
| 	return
 | |
| }
 | |
| 
 | |
| // makeLabels converts annotations to labels and merge them with the given
 | |
| // labels. This is necessary because docker does not support annotations;
 | |
| // we *fake* annotations using labels. Note that docker labels are not
 | |
| // updatable.
 | |
| func makeLabels(labels, annotations map[string]string) map[string]string {
 | |
| 	merged := make(map[string]string)
 | |
| 	for k, v := range labels {
 | |
| 		merged[k] = v
 | |
| 	}
 | |
| 	for k, v := range annotations {
 | |
| 		// Assume there won't be conflict.
 | |
| 		merged[fmt.Sprintf("%s%s", annotationPrefix, k)] = v
 | |
| 	}
 | |
| 	return merged
 | |
| }
 | |
| 
 | |
| // extractLabels converts raw docker labels to the CRI labels and annotations.
 | |
| // It also filters out internal labels used by this shim.
 | |
| func extractLabels(input map[string]string) (map[string]string, map[string]string) {
 | |
| 	labels := make(map[string]string)
 | |
| 	annotations := make(map[string]string)
 | |
| 	for k, v := range input {
 | |
| 		// Check if the key is used internally by the shim.
 | |
| 		internal := false
 | |
| 		for _, internalKey := range internalLabelKeys {
 | |
| 			if k == internalKey {
 | |
| 				internal = true
 | |
| 				break
 | |
| 			}
 | |
| 		}
 | |
| 		if internal {
 | |
| 			continue
 | |
| 		}
 | |
| 
 | |
| 		// Delete the container name label for the sandbox. It is added in the shim,
 | |
| 		// should not be exposed via CRI.
 | |
| 		if k == types.KubernetesContainerNameLabel &&
 | |
| 			input[containerTypeLabelKey] == containerTypeLabelSandbox {
 | |
| 			continue
 | |
| 		}
 | |
| 
 | |
| 		// Check if the label should be treated as an annotation.
 | |
| 		if strings.HasPrefix(k, annotationPrefix) {
 | |
| 			annotations[strings.TrimPrefix(k, annotationPrefix)] = v
 | |
| 			continue
 | |
| 		}
 | |
| 		labels[k] = v
 | |
| 	}
 | |
| 	return labels, annotations
 | |
| }
 | |
| 
 | |
| // generateMountBindings converts the mount list to a list of strings that
 | |
| // can be understood by docker.
 | |
| // Each element in the string is in the form of:
 | |
| // '<HostPath>:<ContainerPath>', or
 | |
| // '<HostPath>:<ContainerPath>:ro', if the path is read only, or
 | |
| // '<HostPath>:<ContainerPath>:Z', if the volume requires SELinux
 | |
| // relabeling and the pod provides an SELinux label
 | |
| func generateMountBindings(mounts []*runtimeapi.Mount) (result []string) {
 | |
| 	for _, m := range mounts {
 | |
| 		bind := fmt.Sprintf("%s:%s", m.HostPath, m.ContainerPath)
 | |
| 		readOnly := m.Readonly
 | |
| 		if readOnly {
 | |
| 			bind += ":ro"
 | |
| 		}
 | |
| 		// Only request relabeling if the pod provides an SELinux context. If the pod
 | |
| 		// does not provide an SELinux context relabeling will label the volume with
 | |
| 		// the container's randomly allocated MCS label. This would restrict access
 | |
| 		// to the volume to the container which mounts it first.
 | |
| 		if m.SelinuxRelabel {
 | |
| 			if readOnly {
 | |
| 				bind += ",Z"
 | |
| 			} else {
 | |
| 				bind += ":Z"
 | |
| 			}
 | |
| 		}
 | |
| 		result = append(result, bind)
 | |
| 	}
 | |
| 	return
 | |
| }
 | |
| 
 | |
| func makePortsAndBindings(pm []*runtimeapi.PortMapping) (map[dockernat.Port]struct{}, map[dockernat.Port][]dockernat.PortBinding) {
 | |
| 	exposedPorts := map[dockernat.Port]struct{}{}
 | |
| 	portBindings := map[dockernat.Port][]dockernat.PortBinding{}
 | |
| 	for _, port := range pm {
 | |
| 		exteriorPort := port.HostPort
 | |
| 		if exteriorPort == 0 {
 | |
| 			// No need to do port binding when HostPort is not specified
 | |
| 			continue
 | |
| 		}
 | |
| 		interiorPort := port.ContainerPort
 | |
| 		// Some of this port stuff is under-documented voodoo.
 | |
| 		// See http://stackoverflow.com/questions/20428302/binding-a-port-to-a-host-interface-using-the-rest-api
 | |
| 		var protocol string
 | |
| 		switch port.Protocol {
 | |
| 		case runtimeapi.Protocol_UDP:
 | |
| 			protocol = "/udp"
 | |
| 		case runtimeapi.Protocol_TCP:
 | |
| 			protocol = "/tcp"
 | |
| 		default:
 | |
| 			glog.Warningf("Unknown protocol %q: defaulting to TCP", port.Protocol)
 | |
| 			protocol = "/tcp"
 | |
| 		}
 | |
| 
 | |
| 		dockerPort := dockernat.Port(strconv.Itoa(int(interiorPort)) + protocol)
 | |
| 		exposedPorts[dockerPort] = struct{}{}
 | |
| 
 | |
| 		hostBinding := dockernat.PortBinding{
 | |
| 			HostPort: strconv.Itoa(int(exteriorPort)),
 | |
| 			HostIP:   port.HostIp,
 | |
| 		}
 | |
| 
 | |
| 		// Allow multiple host ports bind to same docker port
 | |
| 		if existedBindings, ok := portBindings[dockerPort]; ok {
 | |
| 			// If a docker port already map to a host port, just append the host ports
 | |
| 			portBindings[dockerPort] = append(existedBindings, hostBinding)
 | |
| 		} else {
 | |
| 			// Otherwise, it's fresh new port binding
 | |
| 			portBindings[dockerPort] = []dockernat.PortBinding{
 | |
| 				hostBinding,
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 	return exposedPorts, portBindings
 | |
| }
 | |
| 
 | |
| func getSeccompDockerOpts(annotations map[string]string, ctrName, profileRoot string) ([]dockerOpt, error) {
 | |
| 	profile, profileOK := annotations[v1.SeccompContainerAnnotationKeyPrefix+ctrName]
 | |
| 	if !profileOK {
 | |
| 		// try the pod profile
 | |
| 		profile, profileOK = annotations[v1.SeccompPodAnnotationKey]
 | |
| 		if !profileOK {
 | |
| 			// return early the default
 | |
| 			return defaultSeccompOpt, nil
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if profile == "unconfined" {
 | |
| 		// return early the default
 | |
| 		return defaultSeccompOpt, nil
 | |
| 	}
 | |
| 
 | |
| 	if profile == "docker/default" {
 | |
| 		// return nil so docker will load the default seccomp profile
 | |
| 		return nil, nil
 | |
| 	}
 | |
| 
 | |
| 	if !strings.HasPrefix(profile, "localhost/") {
 | |
| 		return nil, fmt.Errorf("unknown seccomp profile option: %s", profile)
 | |
| 	}
 | |
| 
 | |
| 	name := strings.TrimPrefix(profile, "localhost/") // by pod annotation validation, name is a valid subpath
 | |
| 	fname := filepath.Join(profileRoot, filepath.FromSlash(name))
 | |
| 	file, err := ioutil.ReadFile(fname)
 | |
| 	if err != nil {
 | |
| 		return nil, fmt.Errorf("cannot load seccomp profile %q: %v", name, err)
 | |
| 	}
 | |
| 
 | |
| 	b := bytes.NewBuffer(nil)
 | |
| 	if err := json.Compact(b, file); err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 	// Rather than the full profile, just put the filename & md5sum in the event log.
 | |
| 	msg := fmt.Sprintf("%s(md5:%x)", name, md5.Sum(file))
 | |
| 
 | |
| 	return []dockerOpt{{"seccomp", b.String(), msg}}, nil
 | |
| }
 | |
| 
 | |
| // getSeccompSecurityOpts gets container seccomp options from container and sandbox
 | |
| // config, currently from sandbox annotations.
 | |
| // It is an experimental feature and may be promoted to official runtime api in the future.
 | |
| func getSeccompSecurityOpts(containerName string, sandboxConfig *runtimeapi.PodSandboxConfig, seccompProfileRoot string, separator rune) ([]string, error) {
 | |
| 	seccompOpts, err := getSeccompDockerOpts(sandboxConfig.GetAnnotations(), containerName, seccompProfileRoot)
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 	return fmtDockerOpts(seccompOpts, separator), nil
 | |
| }
 | |
| 
 | |
| // getApparmorSecurityOpts gets apparmor options from container config.
 | |
| func getApparmorSecurityOpts(sc *runtimeapi.LinuxContainerSecurityContext, separator rune) ([]string, error) {
 | |
| 	if sc == nil || sc.ApparmorProfile == "" {
 | |
| 		return nil, nil
 | |
| 	}
 | |
| 
 | |
| 	appArmorOpts, err := getAppArmorOpts(sc.ApparmorProfile)
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 
 | |
| 	fmtOpts := fmtDockerOpts(appArmorOpts, separator)
 | |
| 	return fmtOpts, nil
 | |
| }
 | |
| 
 | |
| func getNetworkNamespace(c *dockertypes.ContainerJSON) string {
 | |
| 	if c.State.Pid == 0 {
 | |
| 		// Docker reports pid 0 for an exited container. We can't use it to
 | |
| 		// check the network namespace, so return an empty string instead.
 | |
| 		glog.V(4).Infof("Cannot find network namespace for the terminated container %q", c.ID)
 | |
| 		return ""
 | |
| 	}
 | |
| 	return fmt.Sprintf(dockerNetNSFmt, c.State.Pid)
 | |
| }
 | |
| 
 | |
| // dockerFilter wraps around dockerfilters.Args and provides methods to modify
 | |
| // the filter easily.
 | |
| type dockerFilter struct {
 | |
| 	args *dockerfilters.Args
 | |
| }
 | |
| 
 | |
| func newDockerFilter(args *dockerfilters.Args) *dockerFilter {
 | |
| 	return &dockerFilter{args: args}
 | |
| }
 | |
| 
 | |
| func (f *dockerFilter) Add(key, value string) {
 | |
| 	f.args.Add(key, value)
 | |
| }
 | |
| 
 | |
| func (f *dockerFilter) AddLabel(key, value string) {
 | |
| 	f.Add("label", fmt.Sprintf("%s=%s", key, value))
 | |
| }
 | |
| 
 | |
| // parseUserFromImageUser splits the user out of an user:group string.
 | |
| func parseUserFromImageUser(id string) string {
 | |
| 	if id == "" {
 | |
| 		return id
 | |
| 	}
 | |
| 	// split instances where the id may contain user:group
 | |
| 	if strings.Contains(id, ":") {
 | |
| 		return strings.Split(id, ":")[0]
 | |
| 	}
 | |
| 	// no group, just return the id
 | |
| 	return id
 | |
| }
 | |
| 
 | |
| // getUserFromImageUser gets uid or user name of the image user.
 | |
| // If user is numeric, it will be treated as uid; or else, it is treated as user name.
 | |
| func getUserFromImageUser(imageUser string) (*int64, string) {
 | |
| 	user := parseUserFromImageUser(imageUser)
 | |
| 	// return both nil if user is not specified in the image.
 | |
| 	if user == "" {
 | |
| 		return nil, ""
 | |
| 	}
 | |
| 	// user could be either uid or user name. Try to interpret as numeric uid.
 | |
| 	uid, err := strconv.ParseInt(user, 10, 64)
 | |
| 	if err != nil {
 | |
| 		// If user is non numeric, assume it's user name.
 | |
| 		return nil, user
 | |
| 	}
 | |
| 	// If user is a numeric uid.
 | |
| 	return &uid, ""
 | |
| }
 | |
| 
 | |
| // See #33189. If the previous attempt to create a sandbox container name FOO
 | |
| // failed due to "device or resource busy", it is possible that docker did
 | |
| // not clean up properly and has inconsistent internal state. Docker would
 | |
| // not report the existence of FOO, but would complain if user wants to
 | |
| // create a new container named FOO. To work around this, we parse the error
 | |
| // message to identify failure caused by naming conflict, and try to remove
 | |
| // the old container FOO.
 | |
| // See #40443. Sometimes even removal may fail with "no such container" error.
 | |
| // In that case we have to create the container with a randomized name.
 | |
| // TODO(random-liu): Remove this work around after docker 1.11 is deprecated.
 | |
| // TODO(#33189): Monitor the tests to see if the fix is sufficient.
 | |
| func recoverFromCreationConflictIfNeeded(client libdocker.Interface, createConfig dockertypes.ContainerCreateConfig, err error) (*dockertypes.ContainerCreateResponse, error) {
 | |
| 	matches := conflictRE.FindStringSubmatch(err.Error())
 | |
| 	if len(matches) != 2 {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 
 | |
| 	id := matches[1]
 | |
| 	glog.Warningf("Unable to create pod sandbox due to conflict. Attempting to remove sandbox %q", id)
 | |
| 	if rmErr := client.RemoveContainer(id, dockertypes.ContainerRemoveOptions{RemoveVolumes: true}); rmErr == nil {
 | |
| 		glog.V(2).Infof("Successfully removed conflicting container %q", id)
 | |
| 		return nil, err
 | |
| 	} else {
 | |
| 		glog.Errorf("Failed to remove the conflicting container %q: %v", id, rmErr)
 | |
| 		// Return if the error is not container not found error.
 | |
| 		if !libdocker.IsContainerNotFoundError(rmErr) {
 | |
| 			return nil, err
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	// randomize the name to avoid conflict.
 | |
| 	createConfig.Name = randomizeName(createConfig.Name)
 | |
| 	glog.V(2).Infof("Create the container with randomized name %s", createConfig.Name)
 | |
| 	return client.CreateContainer(createConfig)
 | |
| }
 | |
| 
 | |
| // getSecurityOptSeparator returns the security option separator based on the
 | |
| // docker API version.
 | |
| // TODO: Remove this function along with the relevant code when we no longer
 | |
| // need to support docker 1.10.
 | |
| func getSecurityOptSeparator(v *semver.Version) rune {
 | |
| 	switch v.Compare(optsSeparatorChangeVersion) {
 | |
| 	case -1:
 | |
| 		// Current version is less than the API change version; use the old
 | |
| 		// separator.
 | |
| 		return securityOptSeparatorOld
 | |
| 	default:
 | |
| 		return securityOptSeparatorNew
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // ensureSandboxImageExists pulls the sandbox image when it's not present.
 | |
| func ensureSandboxImageExists(client libdocker.Interface, image string) error {
 | |
| 	dockerCfgSearchPath := []string{"/.docker", filepath.Join(os.Getenv("HOME"), ".docker")}
 | |
| 	return ensureSandboxImageExistsDockerCfg(client, image, dockerCfgSearchPath)
 | |
| }
 | |
| 
 | |
| func ensureSandboxImageExistsDockerCfg(client libdocker.Interface, image string, dockerCfgSearchPath []string) error {
 | |
| 	_, err := client.InspectImageByRef(image)
 | |
| 	if err == nil {
 | |
| 		return nil
 | |
| 	}
 | |
| 	if !libdocker.IsImageNotFoundError(err) {
 | |
| 		return fmt.Errorf("failed to inspect sandbox image %q: %v", image, err)
 | |
| 	}
 | |
| 
 | |
| 	// To support images in private registries, try to read docker config
 | |
| 	authConfig := dockertypes.AuthConfig{}
 | |
| 	keyring := &credentialprovider.BasicDockerKeyring{}
 | |
| 	var cfgLoadErr error
 | |
| 	if cfg, err := credentialprovider.ReadDockerConfigJSONFile(dockerCfgSearchPath); err == nil {
 | |
| 		keyring.Add(cfg)
 | |
| 	} else if cfg, err := credentialprovider.ReadDockercfgFile(dockerCfgSearchPath); err == nil {
 | |
| 		keyring.Add(cfg)
 | |
| 	} else {
 | |
| 		cfgLoadErr = err
 | |
| 	}
 | |
| 	if creds, withCredentials := keyring.Lookup(image); withCredentials {
 | |
| 		// Use the first one that matched our image
 | |
| 		for _, cred := range creds {
 | |
| 			authConfig.Username = cred.Username
 | |
| 			authConfig.Password = cred.Password
 | |
| 			break
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	err = client.PullImage(image, authConfig, dockertypes.ImagePullOptions{})
 | |
| 	if err != nil {
 | |
| 		if cfgLoadErr != nil {
 | |
| 			glog.Warningf("Couldn't load Docker cofig. If sandbox image %q is in a private registry, this will cause further errors. Error: %v", image, cfgLoadErr)
 | |
| 		}
 | |
| 		return fmt.Errorf("unable to pull sandbox image %q: %v", image, err)
 | |
| 	}
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| func getAppArmorOpts(profile string) ([]dockerOpt, error) {
 | |
| 	if profile == "" || profile == apparmor.ProfileRuntimeDefault {
 | |
| 		// The docker applies the default profile by default.
 | |
| 		return nil, nil
 | |
| 	}
 | |
| 
 | |
| 	// Assume validation has already happened.
 | |
| 	profileName := strings.TrimPrefix(profile, apparmor.ProfileNamePrefix)
 | |
| 	return []dockerOpt{{"apparmor", profileName, ""}}, nil
 | |
| }
 | |
| 
 | |
| // fmtDockerOpts formats the docker security options using the given separator.
 | |
| func fmtDockerOpts(opts []dockerOpt, sep rune) []string {
 | |
| 	fmtOpts := make([]string, len(opts))
 | |
| 	for i, opt := range opts {
 | |
| 		fmtOpts[i] = fmt.Sprintf("%s%c%s", opt.key, sep, opt.value)
 | |
| 	}
 | |
| 	return fmtOpts
 | |
| }
 | |
| 
 | |
| type dockerOpt struct {
 | |
| 	// The key-value pair passed to docker.
 | |
| 	key, value string
 | |
| 	// The alternative value to use in log/event messages.
 | |
| 	msg string
 | |
| }
 | |
| 
 | |
| // Expose key/value from  dockerOpt.
 | |
| func (d dockerOpt) GetKV() (string, string) {
 | |
| 	return d.key, d.value
 | |
| }
 |