2014-08-10 02:30:27 +04:00
package daemon
2013-10-05 06:25:15 +04:00
import (
2015-12-11 02:35:10 +03:00
"bytes"
"encoding/json"
"fmt"
"io"
"io/ioutil"
"strings"
"sync"
"github.com/Sirupsen/logrus"
2014-08-10 05:18:32 +04:00
"github.com/docker/docker/opts"
2015-12-11 02:35:10 +03:00
"github.com/docker/docker/pkg/discovery"
2014-08-10 05:18:32 +04:00
flag "github.com/docker/docker/pkg/mflag"
2016-03-09 00:03:37 +03:00
"github.com/docker/docker/registry"
2015-12-11 02:35:10 +03:00
"github.com/imdario/mergo"
2013-10-05 06:25:15 +04:00
)
2016-05-06 07:45:55 +03:00
const (
// defaultMaxConcurrentDownloads is the default value for
// maximum number of downloads that
// may take place at a time for each pull.
defaultMaxConcurrentDownloads = 3
// defaultMaxConcurrentUploads is the default value for
// maximum number of uploads that
// may take place at a time for each push.
defaultMaxConcurrentUploads = 5
2016-06-20 22:14:27 +03:00
// stockRuntimeName is the reserved name/alias used to represent the
// OCI runtime being shipped with the docker daemon package.
stockRuntimeName = "runc"
2016-05-06 07:45:55 +03:00
)
2014-01-30 06:34:43 +04:00
const (
2014-02-01 15:38:39 +04:00
defaultNetworkMtu = 1500
2014-09-16 21:37:50 +04:00
disableNetworkBridge = "none"
2014-01-30 06:34:43 +04:00
)
2016-02-02 22:33:41 +03:00
// flatOptions contains configuration keys
// that MUST NOT be parsed as deep structures.
// Use this to differentiate these options
// with others like the ones in CommonTLSOptions.
var flatOptions = map [ string ] bool {
"cluster-store-opts" : true ,
"log-opts" : true ,
2016-05-24 00:49:50 +03:00
"runtimes" : true ,
2016-02-02 22:33:41 +03:00
}
2015-12-11 02:35:10 +03:00
// LogConfig represents the default log configuration.
// It includes json tags to deserialize configuration from a file
2016-03-28 13:57:55 +03:00
// using the same names that the flags in the command line use.
2015-12-11 02:35:10 +03:00
type LogConfig struct {
Type string ` json:"log-driver,omitempty" `
Config map [ string ] string ` json:"log-opts,omitempty" `
}
2016-03-28 21:55:20 +03:00
// commonBridgeConfig stores all the platform-common bridge driver specific
// configuration.
type commonBridgeConfig struct {
Iface string ` json:"bridge,omitempty" `
FixedCIDR string ` json:"fixed-cidr,omitempty" `
}
2015-12-11 02:35:10 +03:00
// CommonTLSOptions defines TLS configuration for the daemon server.
// It includes json tags to deserialize configuration from a file
2016-03-28 13:57:55 +03:00
// using the same names that the flags in the command line use.
2015-12-11 02:35:10 +03:00
type CommonTLSOptions struct {
CAFile string ` json:"tlscacert,omitempty" `
CertFile string ` json:"tlscert,omitempty" `
KeyFile string ` json:"tlskey,omitempty" `
}
2016-03-28 13:57:55 +03:00
// CommonConfig defines the configuration of a docker daemon which is
2015-04-25 01:36:11 +03:00
// common across platforms.
2015-12-11 02:35:10 +03:00
// It includes json tags to deserialize configuration from a file
2016-03-28 13:57:55 +03:00
// using the same names that the flags in the command line use.
2015-04-25 01:36:11 +03:00
type CommonConfig struct {
2015-12-11 02:35:10 +03:00
AuthorizationPlugins [ ] string ` json:"authorization-plugins,omitempty" ` // AuthorizationPlugins holds list of authorization plugins
AutoRestart bool ` json:"-" `
Context map [ string ] [ ] string ` json:"-" `
DisableBridge bool ` json:"-" `
DNS [ ] string ` json:"dns,omitempty" `
DNSOptions [ ] string ` json:"dns-opts,omitempty" `
DNSSearch [ ] string ` json:"dns-search,omitempty" `
ExecOptions [ ] string ` json:"exec-opts,omitempty" `
GraphDriver string ` json:"storage-driver,omitempty" `
GraphOptions [ ] string ` json:"storage-opts,omitempty" `
Labels [ ] string ` json:"labels,omitempty" `
Mtu int ` json:"mtu,omitempty" `
Pidfile string ` json:"pidfile,omitempty" `
2015-12-13 13:10:41 +03:00
RawLogs bool ` json:"raw-logs,omitempty" `
2015-12-11 02:35:10 +03:00
Root string ` json:"graph,omitempty" `
2016-01-31 05:45:49 +03:00
SocketGroup string ` json:"group,omitempty" `
2015-12-11 02:35:10 +03:00
TrustKeyPath string ` json:"-" `
2016-05-16 06:18:48 +03:00
CorsHeaders string ` json:"api-cors-header,omitempty" `
2016-03-28 17:57:39 +03:00
EnableCors bool ` json:"api-enable-cors,omitempty" `
2016-07-27 18:30:15 +03:00
// LiveRestoreEnabled determines whether we should keep containers
// alive upon daemon shutdown/start
LiveRestoreEnabled bool ` json:"live-restore,omitempty" `
2015-09-11 02:12:00 +03:00
// ClusterStore is the storage backend used for the cluster information. It is used by both
// multihost networking (to store networks and endpoints information) and by the node discovery
// mechanism.
2015-12-11 02:35:10 +03:00
ClusterStore string ` json:"cluster-store,omitempty" `
2015-09-11 02:12:00 +03:00
2015-09-29 02:22:57 +03:00
// ClusterOpts is used to pass options to the discovery package for tuning libkv settings, such
// as TLS configuration settings.
2015-12-11 02:35:10 +03:00
ClusterOpts map [ string ] string ` json:"cluster-store-opts,omitempty" `
2015-09-29 02:22:57 +03:00
2015-09-11 02:12:00 +03:00
// ClusterAdvertise is the network endpoint that the Engine advertises for the purpose of node
// discovery. This should be a 'host:port' combination on which that daemon instance is
// reachable by other hosts.
2015-12-11 02:35:10 +03:00
ClusterAdvertise string ` json:"cluster-advertise,omitempty" `
2016-05-06 07:45:55 +03:00
// MaxConcurrentDownloads is the maximum number of downloads that
// may take place at a time for each pull.
MaxConcurrentDownloads * int ` json:"max-concurrent-downloads,omitempty" `
// MaxConcurrentUploads is the maximum number of uploads that
// may take place at a time for each push.
MaxConcurrentUploads * int ` json:"max-concurrent-uploads,omitempty" `
2016-01-22 21:14:48 +03:00
Debug bool ` json:"debug,omitempty" `
Hosts [ ] string ` json:"hosts,omitempty" `
LogLevel string ` json:"log-level,omitempty" `
TLS bool ` json:"tls,omitempty" `
TLSVerify bool ` json:"tlsverify,omitempty" `
// Embedded structs that allow config
// deserialization without the full struct.
CommonTLSOptions
2016-07-01 04:07:35 +03:00
// SwarmDefaultAdvertiseAddr is the default host/IP or network interface
// to use if a wildcard address is specified in the ListenAddr value
// given to the /swarm/init endpoint and no advertise address is
// specified.
SwarmDefaultAdvertiseAddr string ` json:"swarm-default-advertise-addr" `
2016-01-22 21:14:48 +03:00
LogConfig
2016-01-26 00:30:33 +03:00
bridgeConfig // bridgeConfig holds bridge network specific configuration.
2016-03-09 00:03:37 +03:00
registry . ServiceOptions
2015-12-11 02:35:10 +03:00
reloadLock sync . Mutex
2016-01-19 22:16:07 +03:00
valuesSet map [ string ] interface { }
2013-10-05 06:25:15 +04:00
}
2013-10-21 20:04:42 +04:00
2015-04-25 01:36:11 +03:00
// InstallCommonFlags adds command-line options to the top-level flag parser for
2014-08-10 05:18:32 +04:00
// the current process.
// Subsequent calls to `flag.Parse` will populate config with values parsed
// from the command-line.
2015-05-05 07:18:28 +03:00
func ( config * Config ) InstallCommonFlags ( cmd * flag . FlagSet , usageFn func ( string ) string ) {
2016-05-06 07:45:55 +03:00
var maxConcurrentDownloads , maxConcurrentUploads int
2016-03-09 00:03:37 +03:00
config . ServiceOptions . InstallCliFlags ( cmd , usageFn )
2016-07-16 02:52:59 +03:00
cmd . Var ( opts . NewNamedListOptsRef ( "storage-opts" , & config . GraphOptions , nil ) , [ ] string { "-storage-opt" } , usageFn ( "Storage driver options" ) )
cmd . Var ( opts . NewNamedListOptsRef ( "authorization-plugins" , & config . AuthorizationPlugins , nil ) , [ ] string { "-authorization-plugin" } , usageFn ( "Authorization plugins to load" ) )
cmd . Var ( opts . NewNamedListOptsRef ( "exec-opts" , & config . ExecOptions , nil ) , [ ] string { "-exec-opt" } , usageFn ( "Runtime execution options" ) )
2015-05-05 07:18:28 +03:00
cmd . StringVar ( & config . Pidfile , [ ] string { "p" , "-pidfile" } , defaultPidFile , usageFn ( "Path to use for daemon PID file" ) )
cmd . StringVar ( & config . Root , [ ] string { "g" , "-graph" } , defaultGraph , usageFn ( "Root of the Docker runtime" ) )
cmd . BoolVar ( & config . AutoRestart , [ ] string { "#r" , "#-restart" } , true , usageFn ( "--restart on the daemon has been deprecated in favor of --restart policies on docker run" ) )
cmd . StringVar ( & config . GraphDriver , [ ] string { "s" , "-storage-driver" } , "" , usageFn ( "Storage driver to use" ) )
cmd . IntVar ( & config . Mtu , [ ] string { "#mtu" , "-mtu" } , 0 , usageFn ( "Set the containers network MTU" ) )
2015-12-13 13:10:41 +03:00
cmd . BoolVar ( & config . RawLogs , [ ] string { "-raw-logs" } , false , usageFn ( "Full timestamps without ANSI coloring" ) )
2014-08-10 05:18:32 +04:00
// FIXME: why the inconsistency between "hosts" and "sockets"?
2015-07-31 00:01:53 +03:00
cmd . Var ( opts . NewListOptsRef ( & config . DNS , opts . ValidateIPAddress ) , [ ] string { "#dns" , "-dns" } , usageFn ( "DNS server to use" ) )
2015-12-11 02:35:10 +03:00
cmd . Var ( opts . NewNamedListOptsRef ( "dns-opts" , & config . DNSOptions , nil ) , [ ] string { "-dns-opt" } , usageFn ( "DNS options to use" ) )
2015-07-31 00:01:53 +03:00
cmd . Var ( opts . NewListOptsRef ( & config . DNSSearch , opts . ValidateDNSSearch ) , [ ] string { "-dns-search" } , usageFn ( "DNS search domains to use" ) )
2015-12-11 02:35:10 +03:00
cmd . Var ( opts . NewNamedListOptsRef ( "labels" , & config . Labels , opts . ValidateLabel ) , [ ] string { "-label" } , usageFn ( "Set key=value labels to the daemon" ) )
2015-05-05 07:18:28 +03:00
cmd . StringVar ( & config . LogConfig . Type , [ ] string { "-log-driver" } , "json-file" , usageFn ( "Default driver for container logs" ) )
2016-07-16 02:52:59 +03:00
cmd . Var ( opts . NewNamedMapOpts ( "log-opts" , config . LogConfig . Config , nil ) , [ ] string { "-log-opt" } , usageFn ( "Default log driver options for containers" ) )
2015-10-26 03:12:22 +03:00
cmd . StringVar ( & config . ClusterAdvertise , [ ] string { "-cluster-advertise" } , "" , usageFn ( "Address or interface name to advertise" ) )
2016-07-16 02:52:59 +03:00
cmd . StringVar ( & config . ClusterStore , [ ] string { "-cluster-store" } , "" , usageFn ( "URL of the distributed storage backend" ) )
2015-12-11 02:35:10 +03:00
cmd . Var ( opts . NewNamedMapOpts ( "cluster-store-opts" , config . ClusterOpts , nil ) , [ ] string { "-cluster-store-opt" } , usageFn ( "Set cluster store options" ) )
2016-03-28 17:57:39 +03:00
cmd . StringVar ( & config . CorsHeaders , [ ] string { "-api-cors-header" } , "" , usageFn ( "Set CORS headers in the remote API" ) )
2016-05-06 07:45:55 +03:00
cmd . IntVar ( & maxConcurrentDownloads , [ ] string { "-max-concurrent-downloads" } , defaultMaxConcurrentDownloads , usageFn ( "Set the max concurrent downloads for each pull" ) )
cmd . IntVar ( & maxConcurrentUploads , [ ] string { "-max-concurrent-uploads" } , defaultMaxConcurrentUploads , usageFn ( "Set the max concurrent uploads for each push" ) )
2016-07-01 04:07:35 +03:00
cmd . StringVar ( & config . SwarmDefaultAdvertiseAddr , [ ] string { "-swarm-default-advertise-addr" } , "" , usageFn ( "Set default address or interface for swarm advertised address" ) )
2016-05-06 07:45:55 +03:00
config . MaxConcurrentDownloads = & maxConcurrentDownloads
config . MaxConcurrentUploads = & maxConcurrentUploads
2015-12-11 02:35:10 +03:00
}
2016-01-19 22:16:07 +03:00
// IsValueSet returns true if a configuration value
// was explicitly set in the configuration file.
func ( config * Config ) IsValueSet ( name string ) bool {
if config . valuesSet == nil {
return false
}
_ , ok := config . valuesSet [ name ]
return ok
}
2015-12-11 02:35:10 +03:00
func parseClusterAdvertiseSettings ( clusterStore , clusterAdvertise string ) ( string , error ) {
if clusterAdvertise == "" {
return "" , errDiscoveryDisabled
}
if clusterStore == "" {
return "" , fmt . Errorf ( "invalid cluster configuration. --cluster-advertise must be accompanied by --cluster-store configuration" )
}
advertise , err := discovery . ParseAdvertise ( clusterAdvertise )
if err != nil {
return "" , fmt . Errorf ( "discovery advertise parsing failed (%v)" , err )
}
return advertise , nil
}
// ReloadConfiguration reads the configuration in the host and reloads the daemon and server.
2016-02-19 00:55:03 +03:00
func ReloadConfiguration ( configFile string , flags * flag . FlagSet , reload func ( * Config ) ) error {
2015-12-11 02:35:10 +03:00
logrus . Infof ( "Got signal to reload configuration, reloading from: %s" , configFile )
newConfig , err := getConflictFreeConfiguration ( configFile , flags )
if err != nil {
2016-02-19 00:55:03 +03:00
return err
2015-12-11 02:35:10 +03:00
}
2016-03-11 11:50:49 +03:00
2016-05-24 00:49:50 +03:00
if err := ValidateConfiguration ( newConfig ) ; err != nil {
2016-03-11 11:50:49 +03:00
return fmt . Errorf ( "file configuration validation failed (%v)" , err )
}
2016-02-19 00:55:03 +03:00
reload ( newConfig )
return nil
}
// boolValue is an interface that boolean value flags implement
// to tell the command line how to make -name equivalent to -name=true.
type boolValue interface {
IsBoolFlag ( ) bool
2015-12-11 02:35:10 +03:00
}
// MergeDaemonConfigurations reads a configuration file,
// loads the file configuration in an isolated structure,
// and merges the configuration provided from flags on top
// if there are no conflicts.
func MergeDaemonConfigurations ( flagsConfig * Config , flags * flag . FlagSet , configFile string ) ( * Config , error ) {
fileConfig , err := getConflictFreeConfiguration ( configFile , flags )
if err != nil {
return nil , err
}
2016-05-24 00:49:50 +03:00
if err := ValidateConfiguration ( fileConfig ) ; err != nil {
2016-03-11 11:50:49 +03:00
return nil , fmt . Errorf ( "file configuration validation failed (%v)" , err )
}
2015-12-11 02:35:10 +03:00
// merge flags configuration on top of the file configuration
if err := mergo . Merge ( fileConfig , flagsConfig ) ; err != nil {
return nil , err
}
2016-05-24 00:49:50 +03:00
// We need to validate again once both fileConfig and flagsConfig
// have been merged
if err := ValidateConfiguration ( fileConfig ) ; err != nil {
return nil , fmt . Errorf ( "file configuration validation failed (%v)" , err )
}
2015-12-11 02:35:10 +03:00
return fileConfig , nil
}
// getConflictFreeConfiguration loads the configuration from a JSON file.
// It compares that configuration with the one provided by the flags,
// and returns an error if there are conflicts.
func getConflictFreeConfiguration ( configFile string , flags * flag . FlagSet ) ( * Config , error ) {
b , err := ioutil . ReadFile ( configFile )
if err != nil {
return nil , err
}
2016-01-19 22:16:07 +03:00
var config Config
2015-12-11 02:35:10 +03:00
var reader io . Reader
if flags != nil {
var jsonConfig map [ string ] interface { }
reader = bytes . NewReader ( b )
if err := json . NewDecoder ( reader ) . Decode ( & jsonConfig ) ; err != nil {
return nil , err
}
2016-01-19 22:16:07 +03:00
configSet := configValuesSet ( jsonConfig )
if err := findConfigurationConflicts ( configSet , flags ) ; err != nil {
2015-12-11 02:35:10 +03:00
return nil , err
}
2016-01-19 22:16:07 +03:00
2016-02-19 00:55:03 +03:00
// Override flag values to make sure the values set in the config file with nullable values, like `false`,
2016-07-03 20:58:11 +03:00
// are not overridden by default truthy values from the flags that were not explicitly set.
2016-02-19 00:55:03 +03:00
// See https://github.com/docker/docker/issues/20289 for an example.
//
// TODO: Rewrite configuration logic to avoid same issue with other nullable values, like numbers.
namedOptions := make ( map [ string ] interface { } )
for key , value := range configSet {
f := flags . Lookup ( "-" + key )
if f == nil { // ignore named flags that don't match
namedOptions [ key ] = value
continue
}
if _ , ok := f . Value . ( boolValue ) ; ok {
f . Value . Set ( fmt . Sprintf ( "%v" , value ) )
}
}
if len ( namedOptions ) > 0 {
// set also default for mergeVal flags that are boolValue at the same time.
flags . VisitAll ( func ( f * flag . Flag ) {
if opt , named := f . Value . ( opts . NamedOption ) ; named {
v , set := namedOptions [ opt . Name ( ) ]
_ , boolean := f . Value . ( boolValue )
if set && boolean {
f . Value . Set ( fmt . Sprintf ( "%v" , v ) )
}
}
} )
}
2016-01-19 22:16:07 +03:00
config . valuesSet = configSet
2015-12-11 02:35:10 +03:00
}
reader = bytes . NewReader ( b )
err = json . NewDecoder ( reader ) . Decode ( & config )
return & config , err
}
2016-01-19 22:16:07 +03:00
// configValuesSet returns the configuration values explicitly set in the file.
func configValuesSet ( config map [ string ] interface { } ) map [ string ] interface { } {
2015-12-11 02:35:10 +03:00
flatten := make ( map [ string ] interface { } )
for k , v := range config {
2016-02-02 22:33:41 +03:00
if m , isMap := v . ( map [ string ] interface { } ) ; isMap && ! flatOptions [ k ] {
2015-12-11 02:35:10 +03:00
for km , vm := range m {
flatten [ km ] = vm
}
2016-02-02 22:33:41 +03:00
continue
2015-12-11 02:35:10 +03:00
}
2016-02-02 22:33:41 +03:00
flatten [ k ] = v
2015-12-11 02:35:10 +03:00
}
2016-01-19 22:16:07 +03:00
return flatten
}
// findConfigurationConflicts iterates over the provided flags searching for
2016-01-21 01:16:49 +03:00
// duplicated configurations and unknown keys. It returns an error with all the conflicts if
2016-01-19 22:16:07 +03:00
// it finds any.
func findConfigurationConflicts ( config map [ string ] interface { } , flags * flag . FlagSet ) error {
2016-01-21 01:16:49 +03:00
// 1. Search keys from the file that we don't recognize as flags.
unknownKeys := make ( map [ string ] interface { } )
for key , value := range config {
flagName := "-" + key
if flag := flags . Lookup ( flagName ) ; flag == nil {
unknownKeys [ key ] = value
}
}
2016-01-22 21:14:48 +03:00
// 2. Discard values that implement NamedOption.
// Their configuration name differs from their flag name, like `labels` and `label`.
2016-02-19 00:55:03 +03:00
if len ( unknownKeys ) > 0 {
unknownNamedConflicts := func ( f * flag . Flag ) {
if namedOption , ok := f . Value . ( opts . NamedOption ) ; ok {
if _ , valid := unknownKeys [ namedOption . Name ( ) ] ; valid {
delete ( unknownKeys , namedOption . Name ( ) )
}
2016-01-21 01:16:49 +03:00
}
}
2016-02-19 00:55:03 +03:00
flags . VisitAll ( unknownNamedConflicts )
2016-01-21 01:16:49 +03:00
}
if len ( unknownKeys ) > 0 {
var unknown [ ] string
for key := range unknownKeys {
unknown = append ( unknown , key )
}
return fmt . Errorf ( "the following directives don't match any configuration option: %s" , strings . Join ( unknown , ", " ) )
}
2015-12-11 02:35:10 +03:00
2016-01-21 01:16:49 +03:00
var conflicts [ ] string
2015-12-11 02:35:10 +03:00
printConflict := func ( name string , flagValue , fileValue interface { } ) string {
return fmt . Sprintf ( "%s: (from flag: %v, from file: %v)" , name , flagValue , fileValue )
}
2016-01-21 01:16:49 +03:00
// 3. Search keys that are present as a flag and as a file option.
duplicatedConflicts := func ( f * flag . Flag ) {
2015-12-11 02:35:10 +03:00
// search option name in the json configuration payload if the value is a named option
if namedOption , ok := f . Value . ( opts . NamedOption ) ; ok {
2016-01-19 22:16:07 +03:00
if optsValue , ok := config [ namedOption . Name ( ) ] ; ok {
2015-12-11 02:35:10 +03:00
conflicts = append ( conflicts , printConflict ( namedOption . Name ( ) , f . Value . String ( ) , optsValue ) )
}
} else {
// search flag name in the json configuration payload without trailing dashes
for _ , name := range f . Names {
name = strings . TrimLeft ( name , "-" )
2016-01-19 22:16:07 +03:00
if value , ok := config [ name ] ; ok {
2015-12-11 02:35:10 +03:00
conflicts = append ( conflicts , printConflict ( name , f . Value . String ( ) , value ) )
break
}
}
}
}
2016-01-21 01:16:49 +03:00
flags . Visit ( duplicatedConflicts )
2015-12-11 02:35:10 +03:00
if len ( conflicts ) > 0 {
return fmt . Errorf ( "the following directives are specified both as a flag and in the configuration file: %s" , strings . Join ( conflicts , ", " ) )
}
return nil
2014-08-10 05:18:32 +04:00
}
2016-03-11 11:50:49 +03:00
2016-05-24 00:49:50 +03:00
// ValidateConfiguration validates some specific configs.
2016-05-06 07:45:55 +03:00
// such as config.DNS, config.Labels, config.DNSSearch,
// as well as config.MaxConcurrentDownloads, config.MaxConcurrentUploads.
2016-05-24 00:49:50 +03:00
func ValidateConfiguration ( config * Config ) error {
2016-03-11 11:50:49 +03:00
// validate DNS
for _ , dns := range config . DNS {
if _ , err := opts . ValidateIPAddress ( dns ) ; err != nil {
return err
}
}
// validate DNSSearch
for _ , dnsSearch := range config . DNSSearch {
if _ , err := opts . ValidateDNSSearch ( dnsSearch ) ; err != nil {
return err
}
}
// validate Labels
for _ , label := range config . Labels {
if _ , err := opts . ValidateLabel ( label ) ; err != nil {
return err
}
}
2016-05-06 07:45:55 +03:00
// validate MaxConcurrentDownloads
if config . IsValueSet ( "max-concurrent-downloads" ) && config . MaxConcurrentDownloads != nil && * config . MaxConcurrentDownloads < 0 {
return fmt . Errorf ( "invalid max concurrent downloads: %d" , * config . MaxConcurrentDownloads )
}
// validate MaxConcurrentUploads
if config . IsValueSet ( "max-concurrent-uploads" ) && config . MaxConcurrentUploads != nil && * config . MaxConcurrentUploads < 0 {
return fmt . Errorf ( "invalid max concurrent uploads: %d" , * config . MaxConcurrentUploads )
}
2016-05-24 00:49:50 +03:00
// validate that "default" runtime is not reset
if runtimes := config . GetAllRuntimes ( ) ; len ( runtimes ) > 0 {
2016-06-20 22:14:27 +03:00
if _ , ok := runtimes [ stockRuntimeName ] ; ok {
return fmt . Errorf ( "runtime name '%s' is reserved" , stockRuntimeName )
2016-05-24 00:49:50 +03:00
}
}
2016-06-20 22:14:27 +03:00
if defaultRuntime := config . GetDefaultRuntimeName ( ) ; defaultRuntime != "" && defaultRuntime != stockRuntimeName {
2016-05-24 00:49:50 +03:00
runtimes := config . GetAllRuntimes ( )
if _ , ok := runtimes [ defaultRuntime ] ; ! ok {
return fmt . Errorf ( "specified default runtime '%s' does not exist" , defaultRuntime )
}
}
2016-03-11 11:50:49 +03:00
return nil
}