| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284 | /*   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 ecsimport (	"context"	"fmt"	"github.com/aws/aws-sdk-go/service/elbv2"	"github.com/awslabs/goformation/v4/cloudformation/ec2"	"github.com/awslabs/goformation/v4/cloudformation/elasticloadbalancingv2"	"github.com/awslabs/goformation/v4/cloudformation"	"github.com/awslabs/goformation/v4/cloudformation/ecs"	"github.com/compose-spec/compose-go/types"	"github.com/sirupsen/logrus")// awsResources hold the AWS component being used or created to support services definitiontype awsResources struct {	vpc              string	subnets          []string	cluster          string	loadBalancer     string	loadBalancerType string	securityGroups   map[string]string}func (r *awsResources) serviceSecurityGroups(service types.ServiceConfig) []string {	var groups []string	for net := range service.Networks {		groups = append(groups, r.securityGroups[net])	}	return groups}func (r *awsResources) allSecurityGroups() []string {	var securityGroups []string	for _, r := range r.securityGroups {		securityGroups = append(securityGroups, r)	}	return securityGroups}// parse look into compose project for configured resource to use, and check they are validfunc (b *ecsAPIService) parse(ctx context.Context, project *types.Project) (awsResources, error) {	r := awsResources{}	var err error	r.cluster, err = b.parseClusterExtension(ctx, project)	if err != nil {		return r, err	}	r.vpc, r.subnets, err = b.parseVPCExtension(ctx, project)	if err != nil {		return r, err	}	r.loadBalancer, r.loadBalancerType, err = b.parseLoadBalancerExtension(ctx, project)	if err != nil {		return r, err	}	r.securityGroups, err = b.parseSecurityGroupExtension(ctx, project)	if err != nil {		return r, err	}	return r, nil}func (b *ecsAPIService) parseClusterExtension(ctx context.Context, project *types.Project) (string, error) {	if x, ok := project.Extensions[extensionCluster]; ok {		cluster := x.(string)		ok, err := b.SDK.ClusterExists(ctx, cluster)		if err != nil {			return "", err		}		if !ok {			return "", fmt.Errorf("cluster does not exist: %s", cluster)		}		return cluster, nil	}	return "", nil}func (b *ecsAPIService) parseVPCExtension(ctx context.Context, project *types.Project) (string, []string, error) {	var vpc string	if x, ok := project.Extensions[extensionVPC]; ok {		vpc = x.(string)		err := b.SDK.CheckVPC(ctx, vpc)		if err != nil {			return "", nil, err		}	} else {		defaultVPC, err := b.SDK.GetDefaultVPC(ctx)		if err != nil {			return "", nil, err		}		vpc = defaultVPC	}	subNets, err := b.SDK.GetSubNets(ctx, vpc)	if err != nil {		return "", nil, err	}	if len(subNets) < 2 {		return "", nil, fmt.Errorf("VPC %s should have at least 2 associated subnets in different availability zones", vpc)	}	return vpc, subNets, nil}func (b *ecsAPIService) parseLoadBalancerExtension(ctx context.Context, project *types.Project) (string, string, error) {	if x, ok := project.Extensions[extensionLoadBalancer]; ok {		loadBalancer := x.(string)		loadBalancerType, err := b.SDK.LoadBalancerType(ctx, loadBalancer)		if err != nil {			return "", "", err		}		required := getRequiredLoadBalancerType(project)		if loadBalancerType != required {			return "", "", fmt.Errorf("load balancer %s is of type %s, project require a %s", loadBalancer, loadBalancerType, required)		}		return loadBalancer, loadBalancerType, nil	}	return "", "", nil}func (b *ecsAPIService) parseSecurityGroupExtension(ctx context.Context, project *types.Project) (map[string]string, error) {	securityGroups := make(map[string]string, len(project.Networks))	for name, net := range project.Networks {		var sg string		if net.External.External {			sg = net.Name		}		if x, ok := net.Extensions[extensionSecurityGroup]; ok {			logrus.Warn("to use an existing security-group, use `network.external` and `network.name` in your compose file")			logrus.Debugf("Security Group for network %q set by user to %q", net.Name, x)			sg = x.(string)		}		exists, err := b.SDK.SecurityGroupExists(ctx, sg)		if err != nil {			return nil, err		}		if !exists {			return nil, fmt.Errorf("security group %s doesn't exist", sg)		}		securityGroups[name] = sg	}	return securityGroups, nil}// ensureResources create required resources in template if not yet definedfunc (b *ecsAPIService) ensureResources(resources *awsResources, project *types.Project, template *cloudformation.Template) {	b.ensureCluster(resources, project, template)	b.ensureNetworks(resources, project, template)	b.ensureLoadBalancer(resources, project, template)}func (b *ecsAPIService) ensureCluster(r *awsResources, project *types.Project, template *cloudformation.Template) {	if r.cluster != "" {		return	}	template.Resources["Cluster"] = &ecs.Cluster{		ClusterName: project.Name,		Tags:        projectTags(project),	}	r.cluster = cloudformation.Ref("Cluster")}func (b *ecsAPIService) ensureNetworks(r *awsResources, project *types.Project, template *cloudformation.Template) {	if r.securityGroups == nil {		r.securityGroups = make(map[string]string, len(project.Networks))	}	for name, net := range project.Networks {		securityGroup := networkResourceName(name)		template.Resources[securityGroup] = &ec2.SecurityGroup{			GroupDescription: fmt.Sprintf("%s Security Group for %s network", project.Name, name),			GroupName:        securityGroup,			VpcId:            r.vpc,			Tags:             networkTags(project, net),		}		ingress := securityGroup + "Ingress"		template.Resources[ingress] = &ec2.SecurityGroupIngress{			Description:           fmt.Sprintf("Allow communication within network %s", name),			IpProtocol:            allProtocols,			GroupId:               cloudformation.Ref(securityGroup),			SourceSecurityGroupId: cloudformation.Ref(securityGroup),		}		r.securityGroups[name] = cloudformation.Ref(securityGroup)	}}func (b *ecsAPIService) ensureLoadBalancer(r *awsResources, project *types.Project, template *cloudformation.Template) {	if r.loadBalancer != "" {		return	}	if allServices(project.Services, func(it types.ServiceConfig) bool {		return len(it.Ports) == 0	}) {		logrus.Debug("Application does not expose any public port, so no need for a LoadBalancer")		return	}	balancerType := getRequiredLoadBalancerType(project)	template.Resources["LoadBalancer"] = &elasticloadbalancingv2.LoadBalancer{		Scheme:         elbv2.LoadBalancerSchemeEnumInternetFacing,		SecurityGroups: r.getLoadBalancerSecurityGroups(project),		Subnets:        r.subnets,		Tags:           projectTags(project),		Type:           balancerType,	}	r.loadBalancer = cloudformation.Ref("LoadBalancer")	r.loadBalancerType = balancerType}func (r *awsResources) getLoadBalancerSecurityGroups(project *types.Project) []string {	securityGroups := []string{}	for name, network := range project.Networks {		if !network.Internal {			securityGroups = append(securityGroups, r.securityGroups[name])		}	}	return securityGroups}func getRequiredLoadBalancerType(project *types.Project) string {	loadBalancerType := elbv2.LoadBalancerTypeEnumNetwork	if allServices(project.Services, func(it types.ServiceConfig) bool {		return allPorts(it.Ports, portIsHTTP)	}) {		loadBalancerType = elbv2.LoadBalancerTypeEnumApplication	}	return loadBalancerType}func portIsHTTP(it types.ServicePortConfig) bool {	if v, ok := it.Extensions[extensionProtocol]; ok {		protocol := v.(string)		return protocol == "http" || protocol == "https"	}	return it.Target == 80 || it.Target == 443}// predicate[types.ServiceConfig]type servicePredicate func(it types.ServiceConfig) bool// all[types.ServiceConfig]func allServices(services types.Services, p servicePredicate) bool {	for _, s := range services {		if !p(s) {			return false		}	}	return true}// predicate[types.ServicePortConfig]type portPredicate func(it types.ServicePortConfig) bool// all[types.ServicePortConfig]func allPorts(ports []types.ServicePortConfig, p portPredicate) bool {	for _, s := range ports {		if !p(s) {			return false		}	}	return true}
 |