2014-02-11 23:04:39 -05:00
|
|
|
package runconfig
|
|
|
|
|
|
|
|
import (
|
2015-04-10 20:05:21 -04:00
|
|
|
"encoding/json"
|
|
|
|
"io"
|
2014-05-02 19:59:28 -04:00
|
|
|
"strings"
|
|
|
|
|
2015-06-30 12:41:01 -04:00
|
|
|
"github.com/docker/docker/pkg/nat"
|
2015-09-01 14:26:48 -04:00
|
|
|
"github.com/docker/docker/pkg/stringutils"
|
2015-02-11 14:21:38 -05:00
|
|
|
"github.com/docker/docker/pkg/ulimit"
|
2014-02-11 23:04:39 -05:00
|
|
|
)
|
|
|
|
|
2015-07-25 05:11:45 -04:00
|
|
|
// KeyValuePair is a structure that hold a value for a key.
|
2015-03-29 17:17:23 -04:00
|
|
|
type KeyValuePair struct {
|
|
|
|
Key string
|
|
|
|
Value string
|
|
|
|
}
|
|
|
|
|
2015-07-25 05:11:45 -04:00
|
|
|
// NetworkMode represents the container network stack.
|
2014-05-02 19:59:28 -04:00
|
|
|
type NetworkMode string
|
|
|
|
|
2015-07-25 05:11:45 -04:00
|
|
|
// IpcMode represents the container ipc stack.
|
2014-11-10 16:14:17 -05:00
|
|
|
type IpcMode string
|
|
|
|
|
2015-07-25 05:11:45 -04:00
|
|
|
// IsPrivate indicates whether the container uses it's private ipc stack.
|
2014-11-10 16:14:17 -05:00
|
|
|
func (n IpcMode) IsPrivate() bool {
|
|
|
|
return !(n.IsHost() || n.IsContainer())
|
|
|
|
}
|
|
|
|
|
2015-07-25 05:11:45 -04:00
|
|
|
// IsHost indicates whether the container uses the host's ipc stack.
|
2014-11-10 16:14:17 -05:00
|
|
|
func (n IpcMode) IsHost() bool {
|
|
|
|
return n == "host"
|
|
|
|
}
|
|
|
|
|
2015-07-25 05:11:45 -04:00
|
|
|
// IsContainer indicates whether the container uses a container's ipc stack.
|
2014-11-10 16:14:17 -05:00
|
|
|
func (n IpcMode) IsContainer() bool {
|
|
|
|
parts := strings.SplitN(string(n), ":", 2)
|
|
|
|
return len(parts) > 1 && parts[0] == "container"
|
|
|
|
}
|
|
|
|
|
2015-07-25 05:11:45 -04:00
|
|
|
// Valid indicates whether the ipc stack is valid.
|
2014-11-10 16:14:17 -05:00
|
|
|
func (n IpcMode) Valid() bool {
|
|
|
|
parts := strings.Split(string(n), ":")
|
|
|
|
switch mode := parts[0]; mode {
|
|
|
|
case "", "host":
|
|
|
|
case "container":
|
|
|
|
if len(parts) != 2 || parts[1] == "" {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
2015-07-25 05:11:45 -04:00
|
|
|
// Container returns the name of the container ipc stack is going to be used.
|
2014-11-10 16:14:17 -05:00
|
|
|
func (n IpcMode) Container() string {
|
|
|
|
parts := strings.SplitN(string(n), ":", 2)
|
|
|
|
if len(parts) > 1 {
|
|
|
|
return parts[1]
|
|
|
|
}
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
2015-07-25 05:11:45 -04:00
|
|
|
// UTSMode represents the UTS namespace of the container.
|
2015-05-05 18:32:36 -04:00
|
|
|
type UTSMode string
|
|
|
|
|
2015-07-25 05:11:45 -04:00
|
|
|
// IsPrivate indicates whether the container uses it's private UTS namespace.
|
2015-05-05 18:32:36 -04:00
|
|
|
func (n UTSMode) IsPrivate() bool {
|
|
|
|
return !(n.IsHost())
|
|
|
|
}
|
|
|
|
|
2015-07-25 05:11:45 -04:00
|
|
|
// IsHost indicates whether the container uses the host's UTS namespace.
|
2015-05-05 18:32:36 -04:00
|
|
|
func (n UTSMode) IsHost() bool {
|
|
|
|
return n == "host"
|
|
|
|
}
|
|
|
|
|
2015-07-25 05:11:45 -04:00
|
|
|
// Valid indicates whether the UTS namespace is valid.
|
2015-05-05 18:32:36 -04:00
|
|
|
func (n UTSMode) Valid() bool {
|
|
|
|
parts := strings.Split(string(n), ":")
|
|
|
|
switch mode := parts[0]; mode {
|
|
|
|
case "", "host":
|
|
|
|
default:
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
2015-07-25 05:11:45 -04:00
|
|
|
// PidMode represents the pid stack of the container.
|
2014-11-25 15:10:53 -05:00
|
|
|
type PidMode string
|
|
|
|
|
2015-07-25 05:11:45 -04:00
|
|
|
// IsPrivate indicates whether the container uses it's private pid stack.
|
2014-11-25 15:10:53 -05:00
|
|
|
func (n PidMode) IsPrivate() bool {
|
|
|
|
return !(n.IsHost())
|
|
|
|
}
|
|
|
|
|
2015-07-25 05:11:45 -04:00
|
|
|
// IsHost indicates whether the container uses the host's pid stack.
|
2014-11-25 15:10:53 -05:00
|
|
|
func (n PidMode) IsHost() bool {
|
|
|
|
return n == "host"
|
|
|
|
}
|
|
|
|
|
2015-07-25 05:11:45 -04:00
|
|
|
// Valid indicates whether the pid stack is valid.
|
2014-11-25 15:10:53 -05:00
|
|
|
func (n PidMode) Valid() bool {
|
|
|
|
parts := strings.Split(string(n), ":")
|
|
|
|
switch mode := parts[0]; mode {
|
|
|
|
case "", "host":
|
|
|
|
default:
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
2015-07-25 05:11:45 -04:00
|
|
|
// DeviceMapping represents the device mapping between the host and the container.
|
2014-05-31 00:00:47 -04:00
|
|
|
type DeviceMapping struct {
|
|
|
|
PathOnHost string
|
|
|
|
PathInContainer string
|
|
|
|
CgroupPermissions string
|
|
|
|
}
|
|
|
|
|
2015-07-25 05:11:45 -04:00
|
|
|
// RestartPolicy represents the restart policies of the container.
|
2014-08-04 19:14:43 -04:00
|
|
|
type RestartPolicy struct {
|
|
|
|
Name string
|
|
|
|
MaximumRetryCount int
|
|
|
|
}
|
|
|
|
|
2015-07-25 05:11:45 -04:00
|
|
|
// IsNone indicates whether the container has the "no" restart policy.
|
|
|
|
// This means the container will not automatically restart when exiting.
|
2015-05-16 07:34:09 -04:00
|
|
|
func (rp *RestartPolicy) IsNone() bool {
|
|
|
|
return rp.Name == "no"
|
|
|
|
}
|
|
|
|
|
2015-07-25 05:11:45 -04:00
|
|
|
// IsAlways indicates whether the container has the "always" restart policy.
|
|
|
|
// This means the container will automatically restart regardless of the exit status.
|
2015-05-16 07:34:09 -04:00
|
|
|
func (rp *RestartPolicy) IsAlways() bool {
|
|
|
|
return rp.Name == "always"
|
|
|
|
}
|
|
|
|
|
2015-07-25 05:11:45 -04:00
|
|
|
// IsOnFailure indicates whether the container has the "on-failure" restart policy.
|
|
|
|
// This means the contain will automatically restart of exiting with a non-zero exit status.
|
2015-05-16 07:34:09 -04:00
|
|
|
func (rp *RestartPolicy) IsOnFailure() bool {
|
|
|
|
return rp.Name == "on-failure"
|
|
|
|
}
|
|
|
|
|
2015-08-05 17:09:08 -04:00
|
|
|
// IsUnlessStopped indicates whether the container has the
|
|
|
|
// "unless-stopped" restart policy. This means the container will
|
|
|
|
// automatically restart unless user has put it to stopped state.
|
|
|
|
func (rp *RestartPolicy) IsUnlessStopped() bool {
|
|
|
|
return rp.Name == "unless-stopped"
|
|
|
|
}
|
|
|
|
|
2015-07-25 05:11:45 -04:00
|
|
|
// LogConfig represents the logging configuration of the container.
|
2015-02-04 14:04:58 -05:00
|
|
|
type LogConfig struct {
|
|
|
|
Type string
|
|
|
|
Config map[string]string
|
|
|
|
}
|
|
|
|
|
2015-07-25 05:11:45 -04:00
|
|
|
// LxcConfig represents the specific LXC configuration of the container.
|
2015-04-10 20:05:21 -04:00
|
|
|
type LxcConfig struct {
|
|
|
|
values []KeyValuePair
|
|
|
|
}
|
|
|
|
|
2015-07-25 05:11:45 -04:00
|
|
|
// MarshalJSON marshals (or serializes) the LxcConfig into JSON.
|
2015-04-10 20:05:21 -04:00
|
|
|
func (c *LxcConfig) MarshalJSON() ([]byte, error) {
|
|
|
|
if c == nil {
|
|
|
|
return []byte{}, nil
|
|
|
|
}
|
|
|
|
return json.Marshal(c.Slice())
|
|
|
|
}
|
|
|
|
|
2015-07-25 05:11:45 -04:00
|
|
|
// UnmarshalJSON unmarshals (or deserializes) the specified byte slices from JSON to
|
|
|
|
// a LxcConfig.
|
2015-04-10 20:05:21 -04:00
|
|
|
func (c *LxcConfig) UnmarshalJSON(b []byte) error {
|
|
|
|
if len(b) == 0 {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
var kv []KeyValuePair
|
|
|
|
if err := json.Unmarshal(b, &kv); err != nil {
|
|
|
|
var h map[string]string
|
|
|
|
if err := json.Unmarshal(b, &h); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
for k, v := range h {
|
|
|
|
kv = append(kv, KeyValuePair{k, v})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
c.values = kv
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2015-07-25 05:11:45 -04:00
|
|
|
// Len returns the number of specific lxc configuration.
|
2015-04-10 20:05:21 -04:00
|
|
|
func (c *LxcConfig) Len() int {
|
|
|
|
if c == nil {
|
|
|
|
return 0
|
|
|
|
}
|
|
|
|
return len(c.values)
|
|
|
|
}
|
|
|
|
|
2015-07-25 05:11:45 -04:00
|
|
|
// Slice returns the specific lxc configuration into a slice of KeyValuePair.
|
2015-04-10 20:05:21 -04:00
|
|
|
func (c *LxcConfig) Slice() []KeyValuePair {
|
|
|
|
if c == nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return c.values
|
|
|
|
}
|
|
|
|
|
2015-07-25 05:11:45 -04:00
|
|
|
// NewLxcConfig creates a LxcConfig from the specified slice of KeyValuePair.
|
2015-04-10 20:05:21 -04:00
|
|
|
func NewLxcConfig(values []KeyValuePair) *LxcConfig {
|
|
|
|
return &LxcConfig{values}
|
|
|
|
}
|
|
|
|
|
2015-07-25 05:11:45 -04:00
|
|
|
// HostConfig the non-portable Config structure of a container.
|
|
|
|
// Here, "non-portable" means "dependent of the host we are running on".
|
|
|
|
// Portable information *should* appear in Config.
|
2014-02-11 23:04:39 -05:00
|
|
|
type HostConfig struct {
|
2015-09-23 02:02:45 -04:00
|
|
|
Binds []string // List of volume bindings for this container
|
|
|
|
ContainerIDFile string // File (path) where the containerId is written
|
|
|
|
LxcConf *LxcConfig // Additional lxc configuration
|
|
|
|
Memory int64 // Memory limit (in bytes)
|
|
|
|
MemoryReservation int64 // Memory soft limit (in bytes)
|
|
|
|
MemorySwap int64 // Total memory usage (memory + swap); set `-1` to disable swap
|
|
|
|
KernelMemory int64 // Kernel memory limit (in bytes)
|
|
|
|
CPUShares int64 `json:"CpuShares"` // CPU shares (relative weight vs. other containers)
|
|
|
|
CPUPeriod int64 `json:"CpuPeriod"` // CPU CFS (Completely Fair Scheduler) period
|
|
|
|
CpusetCpus string // CpusetCpus 0-2, 0,1
|
|
|
|
CpusetMems string // CpusetMems 0-2, 0,1
|
|
|
|
CPUQuota int64 `json:"CpuQuota"` // CPU CFS (Completely Fair Scheduler) quota
|
|
|
|
BlkioWeight int64 // Block IO weight (relative weight vs. other containers)
|
|
|
|
OomKillDisable bool // Whether to disable OOM Killer or not
|
|
|
|
MemorySwappiness *int64 // Tuning container memory swappiness behaviour
|
|
|
|
Privileged bool // Is the container in privileged mode
|
|
|
|
PortBindings nat.PortMap // Port mapping between the exposed port (container) and the host
|
|
|
|
Links []string // List of links (in the name:alias form)
|
|
|
|
PublishAllPorts bool // Should docker publish all exposed port for the container
|
|
|
|
DNS []string `json:"Dns"` // List of DNS server to lookup
|
|
|
|
DNSOptions []string `json:"DnsOptions"` // List of DNSOption to look for
|
|
|
|
DNSSearch []string `json:"DnsSearch"` // List of DNSSearch to look for
|
|
|
|
ExtraHosts []string // List of extra hosts
|
|
|
|
VolumesFrom []string // List of volumes to take from other container
|
|
|
|
Devices []DeviceMapping // List of devices to map inside the container
|
|
|
|
NetworkMode NetworkMode // Network namespace to use for the container
|
|
|
|
IpcMode IpcMode // IPC namespace to use for the container
|
|
|
|
PidMode PidMode // PID namespace to use for the container
|
|
|
|
UTSMode UTSMode // UTS namespace to use for the container
|
|
|
|
CapAdd *stringutils.StrSlice // List of kernel capabilities to add to the container
|
|
|
|
CapDrop *stringutils.StrSlice // List of kernel capabilities to remove from the container
|
|
|
|
GroupAdd []string // List of additional groups that the container process will run as
|
|
|
|
RestartPolicy RestartPolicy // Restart policy to be used for the container
|
|
|
|
SecurityOpt []string // List of string values to customize labels for MLS systems, such as SELinux.
|
|
|
|
ReadonlyRootfs bool // Is the container root filesystem in read-only
|
|
|
|
Ulimits []*ulimit.Ulimit // List of ulimits to be set in the container
|
|
|
|
LogConfig LogConfig // Configuration of the logs for this container
|
|
|
|
CgroupParent string // Parent cgroup.
|
|
|
|
ConsoleSize [2]int // Initial console size on Windows
|
|
|
|
VolumeDriver string // Name of the volume driver used to mount volumes
|
2015-07-25 05:11:45 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// DecodeHostConfig creates a HostConfig based on the specified Reader.
|
|
|
|
// It assumes the content of the reader will be JSON, and decodes it.
|
2015-04-10 20:05:21 -04:00
|
|
|
func DecodeHostConfig(src io.Reader) (*HostConfig, error) {
|
|
|
|
decoder := json.NewDecoder(src)
|
|
|
|
|
2015-05-14 10:39:44 -04:00
|
|
|
var w ContainerConfigWrapper
|
2015-04-10 20:05:21 -04:00
|
|
|
if err := decoder.Decode(&w); err != nil {
|
|
|
|
return nil, err
|
2014-07-10 17:51:15 -04:00
|
|
|
}
|
2014-08-04 19:14:43 -04:00
|
|
|
|
2015-07-09 18:12:36 -04:00
|
|
|
hc := w.getHostConfig()
|
2015-04-10 20:05:21 -04:00
|
|
|
return hc, nil
|
2014-02-11 23:04:39 -05:00
|
|
|
}
|
2015-07-09 18:12:36 -04:00
|
|
|
|
|
|
|
// SetDefaultNetModeIfBlank changes the NetworkMode in a HostConfig structure
|
|
|
|
// to default if it is not populated. This ensures backwards compatibility after
|
|
|
|
// the validation of the network mode was moved from the docker CLI to the
|
|
|
|
// docker daemon.
|
|
|
|
func SetDefaultNetModeIfBlank(hc *HostConfig) *HostConfig {
|
|
|
|
if hc != nil {
|
|
|
|
if hc.NetworkMode == NetworkMode("") {
|
|
|
|
hc.NetworkMode = NetworkMode("default")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return hc
|
|
|
|
}
|