2017-02-07 22:33:23 +01:00
|
|
|
package config
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"path"
|
|
|
|
|
|
|
|
"github.com/Sirupsen/logrus"
|
|
|
|
"github.com/docker/libcompose/utils"
|
|
|
|
)
|
|
|
|
|
|
|
|
// MergeServicesV1 merges a v1 compose file into an existing set of service configs
|
2017-07-06 16:28:13 +02:00
|
|
|
func MergeServicesV1(existingServices *ServiceConfigs, environmentLookup EnvironmentLookup, resourceLookup ResourceLookup, file string, datas RawServiceMap, options *ParseOptions) (map[string]*ServiceConfigV1, error) {
|
2017-02-07 22:33:23 +01:00
|
|
|
if options.Validate {
|
|
|
|
if err := validate(datas); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for name, data := range datas {
|
|
|
|
data, err := parseV1(resourceLookup, environmentLookup, file, data, datas, options)
|
|
|
|
if err != nil {
|
|
|
|
logrus.Errorf("Failed to parse service %s: %v", name, err)
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if serviceConfig, ok := existingServices.Get(name); ok {
|
|
|
|
var rawExistingService RawService
|
|
|
|
if err := utils.Convert(serviceConfig, &rawExistingService); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2017-07-06 16:28:13 +02:00
|
|
|
data = mergeConfigV1(rawExistingService, data)
|
2017-02-07 22:33:23 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
datas[name] = data
|
|
|
|
}
|
|
|
|
|
|
|
|
if options.Validate {
|
|
|
|
for name, data := range datas {
|
|
|
|
err := validateServiceConstraints(data, name)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
serviceConfigs := make(map[string]*ServiceConfigV1)
|
|
|
|
if err := utils.Convert(datas, &serviceConfigs); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return serviceConfigs, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func parseV1(resourceLookup ResourceLookup, environmentLookup EnvironmentLookup, inFile string, serviceData RawService, datas RawServiceMap, options *ParseOptions) (RawService, error) {
|
|
|
|
serviceData, err := readEnvFile(resourceLookup, inFile, serviceData)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
serviceData = resolveContextV1(inFile, serviceData)
|
|
|
|
|
|
|
|
value, ok := serviceData["extends"]
|
|
|
|
if !ok {
|
|
|
|
return serviceData, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
mapValue, ok := value.(map[interface{}]interface{})
|
|
|
|
if !ok {
|
|
|
|
return serviceData, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
if resourceLookup == nil {
|
|
|
|
return nil, fmt.Errorf("Can not use extends in file %s no mechanism provided to files", inFile)
|
|
|
|
}
|
|
|
|
|
|
|
|
file := asString(mapValue["file"])
|
|
|
|
service := asString(mapValue["service"])
|
|
|
|
|
|
|
|
if service == "" {
|
|
|
|
return serviceData, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
var baseService RawService
|
|
|
|
|
|
|
|
if file == "" {
|
|
|
|
if serviceData, ok := datas[service]; ok {
|
|
|
|
baseService, err = parseV1(resourceLookup, environmentLookup, inFile, serviceData, datas, options)
|
|
|
|
} else {
|
|
|
|
return nil, fmt.Errorf("Failed to find service %s to extend", service)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
bytes, resolved, err := resourceLookup.Lookup(file, inFile)
|
|
|
|
if err != nil {
|
|
|
|
logrus.Errorf("Failed to lookup file %s: %v", file, err)
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2017-07-06 16:28:13 +02:00
|
|
|
config, err := CreateConfig(bytes)
|
|
|
|
if err != nil {
|
2017-02-07 22:33:23 +01:00
|
|
|
return nil, err
|
|
|
|
}
|
2017-07-06 16:28:13 +02:00
|
|
|
baseRawServices := config.Services
|
2017-02-07 22:33:23 +01:00
|
|
|
|
|
|
|
if options.Interpolate {
|
2017-07-06 16:28:13 +02:00
|
|
|
if err = InterpolateRawServiceMap(&baseRawServices, environmentLookup); err != nil {
|
2017-02-07 22:33:23 +01:00
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if options.Preprocess != nil {
|
|
|
|
var err error
|
|
|
|
baseRawServices, err = options.Preprocess(baseRawServices)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if options.Validate {
|
|
|
|
if err := validate(baseRawServices); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
baseService, ok = baseRawServices[service]
|
|
|
|
if !ok {
|
|
|
|
return nil, fmt.Errorf("Failed to find service %s in file %s", service, file)
|
|
|
|
}
|
|
|
|
|
|
|
|
baseService, err = parseV1(resourceLookup, environmentLookup, resolved, baseService, baseRawServices, options)
|
|
|
|
}
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
baseService = clone(baseService)
|
|
|
|
|
|
|
|
logrus.Debugf("Merging %#v, %#v", baseService, serviceData)
|
|
|
|
|
|
|
|
for _, k := range noMerge {
|
|
|
|
if _, ok := baseService[k]; ok {
|
|
|
|
source := file
|
|
|
|
if source == "" {
|
|
|
|
source = inFile
|
|
|
|
}
|
|
|
|
return nil, fmt.Errorf("Cannot extend service '%s' in %s: services with '%s' cannot be extended", service, source, k)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-07-06 16:28:13 +02:00
|
|
|
baseService = mergeConfigV1(baseService, serviceData)
|
2017-02-07 22:33:23 +01:00
|
|
|
|
|
|
|
logrus.Debugf("Merged result %#v", baseService)
|
|
|
|
|
|
|
|
return baseService, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func resolveContextV1(inFile string, serviceData RawService) RawService {
|
|
|
|
context := asString(serviceData["build"])
|
|
|
|
if context == "" {
|
|
|
|
return serviceData
|
|
|
|
}
|
|
|
|
|
|
|
|
if IsValidRemote(context) {
|
|
|
|
return serviceData
|
|
|
|
}
|
|
|
|
|
|
|
|
current := path.Dir(inFile)
|
|
|
|
|
|
|
|
if context == "." {
|
|
|
|
context = current
|
|
|
|
} else {
|
|
|
|
current = path.Join(current, context)
|
|
|
|
}
|
|
|
|
|
|
|
|
serviceData["build"] = current
|
|
|
|
|
|
|
|
return serviceData
|
|
|
|
}
|
2017-07-06 16:28:13 +02:00
|
|
|
|
|
|
|
func mergeConfigV1(baseService, serviceData RawService) RawService {
|
|
|
|
for k, v := range serviceData {
|
|
|
|
// Image and build are mutually exclusive in merge
|
|
|
|
if k == "image" {
|
|
|
|
delete(baseService, "build")
|
|
|
|
} else if k == "build" {
|
|
|
|
delete(baseService, "image")
|
|
|
|
}
|
|
|
|
existing, ok := baseService[k]
|
|
|
|
if ok {
|
|
|
|
baseService[k] = merge(existing, v)
|
|
|
|
} else {
|
|
|
|
baseService[k] = v
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return baseService
|
|
|
|
}
|