mirror of https://github.com/docker/compose.git
560 lines
16 KiB
Go
560 lines
16 KiB
Go
/*
|
||
Copyright 2020 Docker Compose CLI 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 ecs
|
||
|
||
import (
|
||
"encoding/json"
|
||
"fmt"
|
||
"os"
|
||
"path/filepath"
|
||
"sort"
|
||
"strconv"
|
||
"strings"
|
||
"time"
|
||
|
||
"github.com/aws/aws-sdk-go/aws"
|
||
ecsapi "github.com/aws/aws-sdk-go/service/ecs"
|
||
"github.com/awslabs/goformation/v4/cloudformation"
|
||
"github.com/awslabs/goformation/v4/cloudformation/ecs"
|
||
"github.com/compose-spec/compose-go/types"
|
||
"github.com/docker/cli/opts"
|
||
"github.com/joho/godotenv"
|
||
|
||
"github.com/docker/compose-cli/ecs/secrets"
|
||
)
|
||
|
||
const secretsInitContainerImage = "docker/ecs-secrets-sidecar"
|
||
|
||
func convert(project *types.Project, service types.ServiceConfig) (*ecs.TaskDefinition, error) {
|
||
cpu, mem, err := toLimits(service)
|
||
if err != nil {
|
||
return nil, err
|
||
}
|
||
_, memReservation := toContainerReservation(service)
|
||
credential := getRepoCredentials(service)
|
||
|
||
// override resolve.conf search directive to also search <project>.local
|
||
// TODO remove once ECS support hostname-only service discovery
|
||
service.Environment["LOCALDOMAIN"] = aws.String(
|
||
cloudformation.Join("", []string{
|
||
cloudformation.Ref("AWS::Region"),
|
||
".compute.internal",
|
||
fmt.Sprintf(" %s.local", project.Name),
|
||
}))
|
||
|
||
logConfiguration := getLogConfiguration(service, project)
|
||
|
||
var (
|
||
initContainers []ecs.TaskDefinition_ContainerDefinition
|
||
volumes []ecs.TaskDefinition_Volume
|
||
mounts []ecs.TaskDefinition_MountPoint
|
||
)
|
||
if len(service.Secrets) > 0 {
|
||
secretsVolume, secretsMount, secretsSideCar, err := createSecretsSideCar(project, service, logConfiguration)
|
||
if err != nil {
|
||
return nil, err
|
||
}
|
||
initContainers = append(initContainers, secretsSideCar)
|
||
volumes = append(volumes, secretsVolume)
|
||
mounts = append(mounts, secretsMount)
|
||
}
|
||
|
||
var dependencies []ecs.TaskDefinition_ContainerDependency
|
||
for _, c := range initContainers {
|
||
dependencies = append(dependencies, ecs.TaskDefinition_ContainerDependency{
|
||
Condition: ecsapi.ContainerConditionSuccess,
|
||
ContainerName: c.Name,
|
||
})
|
||
}
|
||
|
||
for _, v := range service.Volumes {
|
||
source := project.Volumes[v.Source]
|
||
volumes = append(volumes, ecs.TaskDefinition_Volume{
|
||
EFSVolumeConfiguration: &ecs.TaskDefinition_EFSVolumeConfiguration{
|
||
FilesystemId: source.Name,
|
||
RootDirectory: source.DriverOpts["root_directory"],
|
||
},
|
||
Name: v.Source,
|
||
})
|
||
mounts = append(mounts, ecs.TaskDefinition_MountPoint{
|
||
ContainerPath: v.Target,
|
||
ReadOnly: v.ReadOnly,
|
||
SourceVolume: v.Source,
|
||
})
|
||
}
|
||
|
||
pairs, err := createEnvironment(project, service)
|
||
if err != nil {
|
||
return nil, err
|
||
}
|
||
|
||
var reservations *types.Resource
|
||
if service.Deploy != nil && service.Deploy.Resources.Reservations != nil {
|
||
reservations = service.Deploy.Resources.Reservations
|
||
}
|
||
|
||
containers := append(initContainers, ecs.TaskDefinition_ContainerDefinition{
|
||
Command: service.Command,
|
||
DisableNetworking: service.NetworkMode == "none",
|
||
DependsOnProp: dependencies,
|
||
DnsSearchDomains: service.DNSSearch,
|
||
DnsServers: service.DNS,
|
||
DockerSecurityOptions: service.SecurityOpt,
|
||
EntryPoint: service.Entrypoint,
|
||
Environment: pairs,
|
||
Essential: true,
|
||
ExtraHosts: toHostEntryPtr(service.ExtraHosts),
|
||
FirelensConfiguration: nil,
|
||
HealthCheck: toHealthCheck(service.HealthCheck),
|
||
Hostname: service.Hostname,
|
||
Image: service.Image,
|
||
Interactive: false,
|
||
Links: nil,
|
||
LinuxParameters: toLinuxParameters(service),
|
||
LogConfiguration: logConfiguration,
|
||
MemoryReservation: memReservation,
|
||
MountPoints: mounts,
|
||
Name: service.Name,
|
||
PortMappings: toPortMappings(service.Ports),
|
||
Privileged: service.Privileged,
|
||
PseudoTerminal: service.Tty,
|
||
ReadonlyRootFilesystem: service.ReadOnly,
|
||
RepositoryCredentials: credential,
|
||
ResourceRequirements: toTaskResourceRequirements(reservations),
|
||
StartTimeout: 0,
|
||
StopTimeout: durationToInt(service.StopGracePeriod),
|
||
SystemControls: toSystemControls(service.Sysctls),
|
||
Ulimits: toUlimits(service.Ulimits),
|
||
User: service.User,
|
||
VolumesFrom: nil,
|
||
WorkingDirectory: service.WorkingDir,
|
||
})
|
||
|
||
launchType := ecsapi.LaunchTypeFargate
|
||
if requireEC2(service) {
|
||
launchType = ecsapi.LaunchTypeEc2
|
||
}
|
||
|
||
return &ecs.TaskDefinition{
|
||
ContainerDefinitions: containers,
|
||
Cpu: cpu,
|
||
Family: fmt.Sprintf("%s-%s", project.Name, service.Name),
|
||
IpcMode: service.Ipc,
|
||
Memory: mem,
|
||
NetworkMode: ecsapi.NetworkModeAwsvpc, // FIXME could be set by service.NetworkMode, Fargate only supports network mode ‘awsvpc’.
|
||
PidMode: service.Pid,
|
||
PlacementConstraints: toPlacementConstraints(service.Deploy),
|
||
ProxyConfiguration: nil,
|
||
RequiresCompatibilities: []string{
|
||
launchType,
|
||
},
|
||
Volumes: volumes,
|
||
}, nil
|
||
}
|
||
|
||
func toTaskResourceRequirements(reservations *types.Resource) []ecs.TaskDefinition_ResourceRequirement {
|
||
if reservations == nil {
|
||
return nil
|
||
}
|
||
var requirements []ecs.TaskDefinition_ResourceRequirement
|
||
for _, r := range reservations.GenericResources {
|
||
if r.DiscreteResourceSpec.Kind == "gpus" {
|
||
requirements = append(requirements, ecs.TaskDefinition_ResourceRequirement{
|
||
Type: ecsapi.ResourceTypeGpu,
|
||
Value: fmt.Sprint(r.DiscreteResourceSpec.Value),
|
||
})
|
||
}
|
||
}
|
||
return requirements
|
||
}
|
||
|
||
func createSecretsSideCar(project *types.Project, service types.ServiceConfig, logConfiguration *ecs.TaskDefinition_LogConfiguration) (
|
||
ecs.TaskDefinition_Volume,
|
||
ecs.TaskDefinition_MountPoint,
|
||
ecs.TaskDefinition_ContainerDefinition,
|
||
error) {
|
||
initContainerName := fmt.Sprintf("%s_Secrets_InitContainer", normalizeResourceName(service.Name))
|
||
secretsVolume := ecs.TaskDefinition_Volume{
|
||
Name: "secrets",
|
||
}
|
||
secretsMount := ecs.TaskDefinition_MountPoint{
|
||
ContainerPath: "/run/secrets/",
|
||
ReadOnly: true,
|
||
SourceVolume: "secrets",
|
||
}
|
||
|
||
var (
|
||
args []secrets.Secret
|
||
taskSecrets []ecs.TaskDefinition_Secret
|
||
)
|
||
for _, s := range service.Secrets {
|
||
secretConfig := project.Secrets[s.Source]
|
||
if s.Target == "" {
|
||
s.Target = s.Source
|
||
}
|
||
taskSecrets = append(taskSecrets, ecs.TaskDefinition_Secret{
|
||
Name: s.Target,
|
||
ValueFrom: secretConfig.Name,
|
||
})
|
||
var keys []string
|
||
if ext, ok := secretConfig.Extensions[extensionKeys]; ok {
|
||
if key, ok := ext.(string); ok {
|
||
keys = append(keys, key)
|
||
} else {
|
||
for _, k := range ext.([]interface{}) {
|
||
keys = append(keys, k.(string))
|
||
}
|
||
}
|
||
}
|
||
args = append(args, secrets.Secret{
|
||
Name: s.Target,
|
||
Keys: keys,
|
||
})
|
||
}
|
||
command, err := json.Marshal(args)
|
||
if err != nil {
|
||
return ecs.TaskDefinition_Volume{}, ecs.TaskDefinition_MountPoint{}, ecs.TaskDefinition_ContainerDefinition{}, err
|
||
}
|
||
secretsSideCar := ecs.TaskDefinition_ContainerDefinition{
|
||
Name: initContainerName,
|
||
Image: secretsInitContainerImage,
|
||
Command: []string{string(command)},
|
||
Essential: false, // FIXME this will be ignored, see https://github.com/awslabs/goformation/issues/61#issuecomment-625139607
|
||
LogConfiguration: logConfiguration,
|
||
MountPoints: []ecs.TaskDefinition_MountPoint{
|
||
{
|
||
ContainerPath: "/run/secrets/",
|
||
ReadOnly: false,
|
||
SourceVolume: "secrets",
|
||
},
|
||
},
|
||
Secrets: taskSecrets,
|
||
}
|
||
return secretsVolume, secretsMount, secretsSideCar, nil
|
||
}
|
||
|
||
func createEnvironment(project *types.Project, service types.ServiceConfig) ([]ecs.TaskDefinition_KeyValuePair, error) {
|
||
environment := map[string]*string{}
|
||
for _, f := range service.EnvFile {
|
||
if !filepath.IsAbs(f) {
|
||
f = filepath.Join(project.WorkingDir, f)
|
||
}
|
||
if _, err := os.Stat(f); os.IsNotExist(err) {
|
||
return nil, err
|
||
}
|
||
file, err := os.Open(f)
|
||
if err != nil {
|
||
return nil, err
|
||
}
|
||
defer file.Close() // nolint:errcheck
|
||
|
||
env, err := godotenv.Parse(file)
|
||
if err != nil {
|
||
return nil, err
|
||
}
|
||
for k, v := range env {
|
||
environment[k] = &v
|
||
}
|
||
}
|
||
for k, v := range service.Environment {
|
||
environment[k] = v
|
||
}
|
||
|
||
var pairs []ecs.TaskDefinition_KeyValuePair
|
||
for k, v := range environment {
|
||
name := k
|
||
var value string
|
||
if v != nil {
|
||
value = *v
|
||
}
|
||
pairs = append(pairs, ecs.TaskDefinition_KeyValuePair{
|
||
Name: name,
|
||
Value: value,
|
||
})
|
||
}
|
||
return pairs, nil
|
||
}
|
||
|
||
func getLogConfiguration(service types.ServiceConfig, project *types.Project) *ecs.TaskDefinition_LogConfiguration {
|
||
options := map[string]string{
|
||
"awslogs-region": cloudformation.Ref("AWS::Region"),
|
||
"awslogs-group": cloudformation.Ref("LogGroup"),
|
||
"awslogs-stream-prefix": project.Name,
|
||
}
|
||
if service.Logging != nil {
|
||
for k, v := range service.Logging.Options {
|
||
if strings.HasPrefix(k, "awslogs-") {
|
||
options[k] = v
|
||
}
|
||
}
|
||
}
|
||
logConfiguration := &ecs.TaskDefinition_LogConfiguration{
|
||
LogDriver: ecsapi.LogDriverAwslogs,
|
||
Options: options,
|
||
}
|
||
return logConfiguration
|
||
}
|
||
|
||
func toSystemControls(sysctls types.Mapping) []ecs.TaskDefinition_SystemControl {
|
||
sys := []ecs.TaskDefinition_SystemControl{}
|
||
for k, v := range sysctls {
|
||
sys = append(sys, ecs.TaskDefinition_SystemControl{
|
||
Namespace: k,
|
||
Value: v,
|
||
})
|
||
}
|
||
return sys
|
||
}
|
||
|
||
const miB = 1024 * 1024
|
||
|
||
func toLimits(service types.ServiceConfig) (string, string, error) {
|
||
mem, cpu, err := getConfiguredLimits(service)
|
||
if err != nil {
|
||
return "", "", err
|
||
}
|
||
if requireEC2(service) {
|
||
// just return configured limits expressed in Mb and CPU units
|
||
var cpuLimit, memLimit string
|
||
if cpu > 0 {
|
||
cpuLimit = fmt.Sprint(cpu)
|
||
}
|
||
if mem > 0 {
|
||
memLimit = fmt.Sprint(mem / miB)
|
||
}
|
||
return cpuLimit, memLimit, nil
|
||
}
|
||
|
||
// All possible cpu/mem values for Fargate
|
||
fargateCPUToMem := map[int64][]types.UnitBytes{
|
||
256: {512, 1024, 2048},
|
||
512: {1024, 2048, 3072, 4096},
|
||
1024: {2048, 3072, 4096, 5120, 6144, 7168, 8192},
|
||
2048: {4096, 5120, 6144, 7168, 8192, 9216, 10240, 11264, 12288, 13312, 14336, 15360, 16384},
|
||
4096: {8192, 9216, 10240, 11264, 12288, 13312, 14336, 15360, 16384, 17408, 18432, 19456, 20480, 21504, 22528, 23552, 24576, 25600, 26624, 27648, 28672, 29696, 30720},
|
||
}
|
||
cpuLimit := "256"
|
||
memLimit := "512"
|
||
if mem == 0 && cpu == 0 {
|
||
return cpuLimit, memLimit, nil
|
||
}
|
||
|
||
var cpus []int64
|
||
for k := range fargateCPUToMem {
|
||
cpus = append(cpus, k)
|
||
}
|
||
sort.Slice(cpus, func(i, j int) bool { return cpus[i] < cpus[j] })
|
||
|
||
for _, fargateCPU := range cpus {
|
||
options := fargateCPUToMem[fargateCPU]
|
||
if cpu <= fargateCPU {
|
||
for _, m := range options {
|
||
if mem <= m*miB {
|
||
cpuLimit = strconv.FormatInt(fargateCPU, 10)
|
||
memLimit = strconv.FormatInt(int64(m), 10)
|
||
return cpuLimit, memLimit, nil
|
||
}
|
||
}
|
||
}
|
||
}
|
||
return "", "", fmt.Errorf("the resources requested are not supported by ECS/Fargate")
|
||
}
|
||
|
||
func getConfiguredLimits(service types.ServiceConfig) (types.UnitBytes, int64, error) {
|
||
if service.Deploy == nil {
|
||
return 0, 0, nil
|
||
}
|
||
|
||
limits := service.Deploy.Resources.Limits
|
||
if limits == nil {
|
||
return 0, 0, nil
|
||
}
|
||
|
||
if limits.NanoCPUs == "" {
|
||
return limits.MemoryBytes, 0, nil
|
||
}
|
||
v, err := opts.ParseCPUs(limits.NanoCPUs)
|
||
if err != nil {
|
||
return 0, 0, err
|
||
}
|
||
|
||
return limits.MemoryBytes, v / 1e6, nil
|
||
}
|
||
|
||
func toContainerReservation(service types.ServiceConfig) (string, int) {
|
||
cpuReservation := ".0"
|
||
memReservation := 0
|
||
|
||
if service.Deploy == nil {
|
||
return cpuReservation, memReservation
|
||
}
|
||
|
||
reservations := service.Deploy.Resources.Reservations
|
||
if reservations == nil {
|
||
return cpuReservation, memReservation
|
||
}
|
||
return reservations.NanoCPUs, int(reservations.MemoryBytes / miB)
|
||
}
|
||
|
||
func toPlacementConstraints(deploy *types.DeployConfig) []ecs.TaskDefinition_TaskDefinitionPlacementConstraint {
|
||
if deploy == nil || deploy.Placement.Constraints == nil || len(deploy.Placement.Constraints) == 0 {
|
||
return nil
|
||
}
|
||
pl := []ecs.TaskDefinition_TaskDefinitionPlacementConstraint{}
|
||
for _, c := range deploy.Placement.Constraints {
|
||
pl = append(pl, ecs.TaskDefinition_TaskDefinitionPlacementConstraint{
|
||
Expression: c,
|
||
Type: "",
|
||
})
|
||
}
|
||
return pl
|
||
}
|
||
|
||
func toPortMappings(ports []types.ServicePortConfig) []ecs.TaskDefinition_PortMapping {
|
||
if len(ports) == 0 {
|
||
return nil
|
||
}
|
||
m := []ecs.TaskDefinition_PortMapping{}
|
||
for _, p := range ports {
|
||
m = append(m, ecs.TaskDefinition_PortMapping{
|
||
ContainerPort: int(p.Target),
|
||
HostPort: int(p.Published),
|
||
Protocol: p.Protocol,
|
||
})
|
||
}
|
||
return m
|
||
}
|
||
|
||
func toUlimits(ulimits map[string]*types.UlimitsConfig) []ecs.TaskDefinition_Ulimit {
|
||
if len(ulimits) == 0 {
|
||
return nil
|
||
}
|
||
u := []ecs.TaskDefinition_Ulimit{}
|
||
for k, v := range ulimits {
|
||
u = append(u, ecs.TaskDefinition_Ulimit{
|
||
Name: k,
|
||
SoftLimit: v.Soft,
|
||
HardLimit: v.Hard,
|
||
})
|
||
}
|
||
return u
|
||
}
|
||
|
||
func toLinuxParameters(service types.ServiceConfig) *ecs.TaskDefinition_LinuxParameters {
|
||
return &ecs.TaskDefinition_LinuxParameters{
|
||
Capabilities: toKernelCapabilities(service.CapAdd, service.CapDrop),
|
||
Devices: nil,
|
||
InitProcessEnabled: service.Init != nil && *service.Init,
|
||
MaxSwap: 0,
|
||
// FIXME SharedMemorySize: service.ShmSize,
|
||
Swappiness: 0,
|
||
Tmpfs: toTmpfs(service.Tmpfs),
|
||
}
|
||
}
|
||
|
||
func toTmpfs(tmpfs types.StringList) []ecs.TaskDefinition_Tmpfs {
|
||
if tmpfs == nil || len(tmpfs) == 0 {
|
||
return nil
|
||
}
|
||
o := []ecs.TaskDefinition_Tmpfs{}
|
||
for _, path := range tmpfs {
|
||
o = append(o, ecs.TaskDefinition_Tmpfs{
|
||
ContainerPath: path,
|
||
Size: 100, // size is required on ECS, unlimited by the compose spec
|
||
})
|
||
}
|
||
return o
|
||
}
|
||
|
||
func toKernelCapabilities(add []string, drop []string) *ecs.TaskDefinition_KernelCapabilities {
|
||
if len(add) == 0 && len(drop) == 0 {
|
||
return nil
|
||
}
|
||
return &ecs.TaskDefinition_KernelCapabilities{
|
||
Add: add,
|
||
Drop: drop,
|
||
}
|
||
|
||
}
|
||
|
||
func toHealthCheck(check *types.HealthCheckConfig) *ecs.TaskDefinition_HealthCheck {
|
||
if check == nil {
|
||
return nil
|
||
}
|
||
retries := 0
|
||
if check.Retries != nil {
|
||
retries = int(*check.Retries)
|
||
}
|
||
return &ecs.TaskDefinition_HealthCheck{
|
||
Command: check.Test,
|
||
Interval: durationToInt(check.Interval),
|
||
Retries: retries,
|
||
StartPeriod: durationToInt(check.StartPeriod),
|
||
Timeout: durationToInt(check.Timeout),
|
||
}
|
||
}
|
||
|
||
func durationToInt(interval *types.Duration) int {
|
||
if interval == nil {
|
||
return 0
|
||
}
|
||
v := int(time.Duration(*interval).Seconds())
|
||
return v
|
||
}
|
||
|
||
func toHostEntryPtr(hosts types.HostsList) []ecs.TaskDefinition_HostEntry {
|
||
if hosts == nil || len(hosts) == 0 {
|
||
return nil
|
||
}
|
||
e := []ecs.TaskDefinition_HostEntry{}
|
||
for _, h := range hosts {
|
||
parts := strings.SplitN(h, ":", 2) // FIXME this should be handled by compose-go
|
||
e = append(e, ecs.TaskDefinition_HostEntry{
|
||
Hostname: parts[0],
|
||
IpAddress: parts[1],
|
||
})
|
||
}
|
||
return e
|
||
}
|
||
|
||
func getRepoCredentials(service types.ServiceConfig) *ecs.TaskDefinition_RepositoryCredentials {
|
||
// extract registry and namespace string from image name
|
||
for key, value := range service.Extensions {
|
||
if key == extensionPullCredentials {
|
||
return &ecs.TaskDefinition_RepositoryCredentials{CredentialsParameter: value.(string)}
|
||
}
|
||
}
|
||
return nil
|
||
}
|
||
|
||
func requireEC2(s types.ServiceConfig) bool {
|
||
return gpuRequirements(s) > 0
|
||
}
|
||
|
||
func gpuRequirements(s types.ServiceConfig) int64 {
|
||
if deploy := s.Deploy; deploy != nil {
|
||
if reservations := deploy.Resources.Reservations; reservations != nil {
|
||
for _, resource := range reservations.GenericResources {
|
||
if resource.DiscreteResourceSpec.Kind == "gpus" {
|
||
return resource.DiscreteResourceSpec.Value
|
||
}
|
||
}
|
||
}
|
||
}
|
||
return 0
|
||
}
|