159 lines
4.5 KiB
Go
159 lines
4.5 KiB
Go
//go:build linux && !dockerless
|
|
// +build linux,!dockerless
|
|
|
|
/*
|
|
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 cm
|
|
|
|
import (
|
|
"fmt"
|
|
"io/ioutil"
|
|
"regexp"
|
|
"strconv"
|
|
"time"
|
|
|
|
"github.com/opencontainers/runc/libcontainer/cgroups"
|
|
cgroupfs "github.com/opencontainers/runc/libcontainer/cgroups/fs"
|
|
"github.com/opencontainers/runc/libcontainer/configs"
|
|
utilversion "k8s.io/apimachinery/pkg/util/version"
|
|
"k8s.io/apimachinery/pkg/util/wait"
|
|
"k8s.io/klog/v2"
|
|
kubecm "k8s.io/kubernetes/pkg/kubelet/cm"
|
|
|
|
"k8s.io/kubernetes/pkg/kubelet/dockershim/libdocker"
|
|
)
|
|
|
|
const (
|
|
// The percent of the machine memory capacity. The value is used to calculate
|
|
// docker memory resource container's hardlimit to workaround docker memory
|
|
// leakage issue. Please see kubernetes/issues/9881 for more detail.
|
|
dockerMemoryLimitThresholdPercent = 70
|
|
|
|
// The minimum memory limit allocated to docker container: 150Mi
|
|
minDockerMemoryLimit = 150 * 1024 * 1024
|
|
|
|
// The OOM score adjustment for the docker process (i.e. the docker
|
|
// daemon). Essentially, makes docker very unlikely to experience an oom
|
|
// kill.
|
|
dockerOOMScoreAdj = -999
|
|
)
|
|
|
|
var (
|
|
memoryCapacityRegexp = regexp.MustCompile(`MemTotal:\s*([0-9]+) kB`)
|
|
)
|
|
|
|
// NewContainerManager creates a new instance of ContainerManager
|
|
func NewContainerManager(cgroupsName string, client libdocker.Interface) ContainerManager {
|
|
return &containerManager{
|
|
cgroupsName: cgroupsName,
|
|
client: client,
|
|
}
|
|
}
|
|
|
|
type containerManager struct {
|
|
// Docker client.
|
|
client libdocker.Interface
|
|
// Name of the cgroups.
|
|
cgroupsName string
|
|
// Manager for the cgroups.
|
|
cgroupsManager cgroups.Manager
|
|
}
|
|
|
|
func (m *containerManager) Start() error {
|
|
// TODO: check if the required cgroups are mounted.
|
|
if len(m.cgroupsName) != 0 {
|
|
manager, err := createCgroupManager(m.cgroupsName)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
m.cgroupsManager = manager
|
|
}
|
|
go wait.Until(m.doWork, 5*time.Minute, wait.NeverStop)
|
|
return nil
|
|
}
|
|
|
|
func (m *containerManager) doWork() {
|
|
v, err := m.client.Version()
|
|
if err != nil {
|
|
klog.ErrorS(err, "Unable to get docker version")
|
|
return
|
|
}
|
|
version, err := utilversion.ParseGeneric(v.APIVersion)
|
|
if err != nil {
|
|
klog.ErrorS(err, "Unable to parse docker version", "dockerVersion", v.APIVersion)
|
|
return
|
|
}
|
|
// EnsureDockerInContainer does two things.
|
|
// 1. Ensure processes run in the cgroups if m.cgroupsManager is not nil.
|
|
// 2. Ensure processes have the OOM score applied.
|
|
if err := kubecm.EnsureDockerInContainer(version, dockerOOMScoreAdj, m.cgroupsManager); err != nil {
|
|
klog.ErrorS(err, "Unable to ensure the docker processes run in the desired containers")
|
|
}
|
|
}
|
|
|
|
func createCgroupManager(name string) (cgroups.Manager, error) {
|
|
var memoryLimit uint64
|
|
|
|
memoryCapacity, err := getMemoryCapacity()
|
|
if err != nil {
|
|
klog.ErrorS(err, "Failed to get the memory capacity on machine")
|
|
} else {
|
|
memoryLimit = memoryCapacity * dockerMemoryLimitThresholdPercent / 100
|
|
}
|
|
|
|
if err != nil || memoryLimit < minDockerMemoryLimit {
|
|
memoryLimit = minDockerMemoryLimit
|
|
}
|
|
klog.V(2).InfoS("Configure resource-only container with memory limit", "containerName", name, "memoryLimit", memoryLimit)
|
|
|
|
cg := &configs.Cgroup{
|
|
Parent: "/",
|
|
Name: name,
|
|
Resources: &configs.Resources{
|
|
Memory: int64(memoryLimit),
|
|
MemorySwap: -1,
|
|
SkipDevices: true,
|
|
},
|
|
}
|
|
return cgroupfs.NewManager(cg, nil, false), nil
|
|
}
|
|
|
|
// getMemoryCapacity returns the memory capacity on the machine in bytes.
|
|
func getMemoryCapacity() (uint64, error) {
|
|
out, err := ioutil.ReadFile("/proc/meminfo")
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
return parseCapacity(out, memoryCapacityRegexp)
|
|
}
|
|
|
|
// parseCapacity matches a Regexp in a []byte, returning the resulting value in bytes.
|
|
// Assumes that the value matched by the Regexp is in KB.
|
|
func parseCapacity(b []byte, r *regexp.Regexp) (uint64, error) {
|
|
matches := r.FindSubmatch(b)
|
|
if len(matches) != 2 {
|
|
return 0, fmt.Errorf("failed to match regexp in output: %q", string(b))
|
|
}
|
|
m, err := strconv.ParseUint(string(matches[1]), 10, 64)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
|
|
// Convert to bytes.
|
|
return m * 1024, err
|
|
}
|