sdk.go 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666
  1. /*
  2. Copyright 2020 Docker, Inc.
  3. Licensed under the Apache License, Version 2.0 (the "License");
  4. you may not use this file except in compliance with the License.
  5. You may obtain a copy of the License at
  6. http://www.apache.org/licenses/LICENSE-2.0
  7. Unless required by applicable law or agreed to in writing, software
  8. distributed under the License is distributed on an "AS IS" BASIS,
  9. WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  10. See the License for the specific language governing permissions and
  11. limitations under the License.
  12. */
  13. package ecs
  14. import (
  15. "context"
  16. "fmt"
  17. "strings"
  18. "time"
  19. "github.com/aws/aws-sdk-go/aws/client"
  20. "github.com/docker/compose-cli/api/compose"
  21. "github.com/docker/compose-cli/api/secrets"
  22. "github.com/aws/aws-sdk-go/aws"
  23. "github.com/aws/aws-sdk-go/service/cloudformation"
  24. "github.com/aws/aws-sdk-go/service/cloudformation/cloudformationiface"
  25. "github.com/aws/aws-sdk-go/service/cloudwatchlogs"
  26. "github.com/aws/aws-sdk-go/service/cloudwatchlogs/cloudwatchlogsiface"
  27. "github.com/aws/aws-sdk-go/service/ec2"
  28. "github.com/aws/aws-sdk-go/service/ec2/ec2iface"
  29. "github.com/aws/aws-sdk-go/service/ecs"
  30. "github.com/aws/aws-sdk-go/service/ecs/ecsiface"
  31. "github.com/aws/aws-sdk-go/service/elbv2"
  32. "github.com/aws/aws-sdk-go/service/elbv2/elbv2iface"
  33. "github.com/aws/aws-sdk-go/service/iam"
  34. "github.com/aws/aws-sdk-go/service/iam/iamiface"
  35. "github.com/aws/aws-sdk-go/service/secretsmanager"
  36. "github.com/aws/aws-sdk-go/service/secretsmanager/secretsmanageriface"
  37. cf "github.com/awslabs/goformation/v4/cloudformation"
  38. "github.com/sirupsen/logrus"
  39. )
  40. type sdk struct {
  41. ECS ecsiface.ECSAPI
  42. EC2 ec2iface.EC2API
  43. ELB elbv2iface.ELBV2API
  44. CW cloudwatchlogsiface.CloudWatchLogsAPI
  45. IAM iamiface.IAMAPI
  46. CF cloudformationiface.CloudFormationAPI
  47. SM secretsmanageriface.SecretsManagerAPI
  48. }
  49. func newSDK(sess client.ConfigProvider) sdk {
  50. return sdk{
  51. ECS: ecs.New(sess),
  52. EC2: ec2.New(sess),
  53. ELB: elbv2.New(sess),
  54. CW: cloudwatchlogs.New(sess),
  55. IAM: iam.New(sess),
  56. CF: cloudformation.New(sess),
  57. SM: secretsmanager.New(sess),
  58. }
  59. }
  60. func (s sdk) CheckRequirements(ctx context.Context, region string) error {
  61. settings, err := s.ECS.ListAccountSettingsWithContext(ctx, &ecs.ListAccountSettingsInput{
  62. EffectiveSettings: aws.Bool(true),
  63. Name: aws.String("serviceLongArnFormat"),
  64. })
  65. if err != nil {
  66. return err
  67. }
  68. serviceLongArnFormat := settings.Settings[0].Value
  69. if *serviceLongArnFormat != "enabled" {
  70. return fmt.Errorf("this tool requires the \"new ARN resource ID format\".\n"+
  71. "Check https://%s.console.aws.amazon.com/ecs/home#/settings\n"+
  72. "Learn more: https://aws.amazon.com/blogs/compute/migrating-your-amazon-ecs-deployment-to-the-new-arn-and-resource-id-format-2", region)
  73. }
  74. return nil
  75. }
  76. func (s sdk) ClusterExists(ctx context.Context, name string) (bool, error) {
  77. logrus.Debug("CheckRequirements if cluster was already created: ", name)
  78. clusters, err := s.ECS.DescribeClustersWithContext(ctx, &ecs.DescribeClustersInput{
  79. Clusters: []*string{aws.String(name)},
  80. })
  81. if err != nil {
  82. return false, err
  83. }
  84. return len(clusters.Clusters) > 0, nil
  85. }
  86. func (s sdk) CreateCluster(ctx context.Context, name string) (string, error) {
  87. logrus.Debug("Create cluster ", name)
  88. response, err := s.ECS.CreateClusterWithContext(ctx, &ecs.CreateClusterInput{ClusterName: aws.String(name)})
  89. if err != nil {
  90. return "", err
  91. }
  92. return *response.Cluster.Status, nil
  93. }
  94. func (s sdk) CheckVPC(ctx context.Context, vpcID string) error {
  95. logrus.Debug("CheckRequirements on VPC : ", vpcID)
  96. output, err := s.EC2.DescribeVpcAttributeWithContext(ctx, &ec2.DescribeVpcAttributeInput{
  97. VpcId: aws.String(vpcID),
  98. Attribute: aws.String("enableDnsSupport"),
  99. })
  100. if err != nil {
  101. return err
  102. }
  103. if !*output.EnableDnsSupport.Value {
  104. return fmt.Errorf("VPC %q doesn't have DNS resolution enabled", vpcID)
  105. }
  106. return err
  107. }
  108. func (s sdk) GetDefaultVPC(ctx context.Context) (string, error) {
  109. logrus.Debug("Retrieve default VPC")
  110. vpcs, err := s.EC2.DescribeVpcsWithContext(ctx, &ec2.DescribeVpcsInput{
  111. Filters: []*ec2.Filter{
  112. {
  113. Name: aws.String("isDefault"),
  114. Values: []*string{aws.String("true")},
  115. },
  116. },
  117. })
  118. if err != nil {
  119. return "", err
  120. }
  121. if len(vpcs.Vpcs) == 0 {
  122. return "", fmt.Errorf("account has not default VPC")
  123. }
  124. return *vpcs.Vpcs[0].VpcId, nil
  125. }
  126. func (s sdk) GetSubNets(ctx context.Context, vpcID string) ([]string, error) {
  127. logrus.Debug("Retrieve SubNets")
  128. subnets, err := s.EC2.DescribeSubnetsWithContext(ctx, &ec2.DescribeSubnetsInput{
  129. DryRun: nil,
  130. Filters: []*ec2.Filter{
  131. {
  132. Name: aws.String("vpc-id"),
  133. Values: []*string{aws.String(vpcID)},
  134. },
  135. },
  136. })
  137. if err != nil {
  138. return nil, err
  139. }
  140. ids := []string{}
  141. for _, subnet := range subnets.Subnets {
  142. ids = append(ids, *subnet.SubnetId)
  143. }
  144. return ids, nil
  145. }
  146. func (s sdk) GetRoleArn(ctx context.Context, name string) (string, error) {
  147. role, err := s.IAM.GetRoleWithContext(ctx, &iam.GetRoleInput{
  148. RoleName: aws.String(name),
  149. })
  150. if err != nil {
  151. return "", err
  152. }
  153. return *role.Role.Arn, nil
  154. }
  155. func (s sdk) StackExists(ctx context.Context, name string) (bool, error) {
  156. stacks, err := s.CF.DescribeStacksWithContext(ctx, &cloudformation.DescribeStacksInput{
  157. StackName: aws.String(name),
  158. })
  159. if err != nil {
  160. if strings.HasPrefix(err.Error(), fmt.Sprintf("ValidationError: Stack with id %s does not exist", name)) {
  161. return false, nil
  162. }
  163. return false, nil
  164. }
  165. return len(stacks.Stacks) > 0, nil
  166. }
  167. func (s sdk) CreateStack(ctx context.Context, name string, template *cf.Template, parameters map[string]string) error {
  168. logrus.Debug("Create CloudFormation stack")
  169. json, err := marshall(template)
  170. if err != nil {
  171. return err
  172. }
  173. param := []*cloudformation.Parameter{}
  174. for name, value := range parameters {
  175. param = append(param, &cloudformation.Parameter{
  176. ParameterKey: aws.String(name),
  177. ParameterValue: aws.String(value),
  178. })
  179. }
  180. _, err = s.CF.CreateStackWithContext(ctx, &cloudformation.CreateStackInput{
  181. OnFailure: aws.String("DELETE"),
  182. StackName: aws.String(name),
  183. TemplateBody: aws.String(string(json)),
  184. Parameters: param,
  185. TimeoutInMinutes: nil,
  186. Capabilities: []*string{
  187. aws.String(cloudformation.CapabilityCapabilityIam),
  188. },
  189. Tags: []*cloudformation.Tag{
  190. {
  191. Key: aws.String(compose.ProjectTag),
  192. Value: aws.String(name),
  193. },
  194. },
  195. })
  196. return err
  197. }
  198. func (s sdk) CreateChangeSet(ctx context.Context, name string, template *cf.Template, parameters map[string]string) (string, error) {
  199. logrus.Debug("Create CloudFormation Changeset")
  200. json, err := marshall(template)
  201. if err != nil {
  202. return "", err
  203. }
  204. param := []*cloudformation.Parameter{}
  205. for name := range parameters {
  206. param = append(param, &cloudformation.Parameter{
  207. ParameterKey: aws.String(name),
  208. UsePreviousValue: aws.Bool(true),
  209. })
  210. }
  211. update := fmt.Sprintf("Update%s", time.Now().Format("2006-01-02-15-04-05"))
  212. changeset, err := s.CF.CreateChangeSetWithContext(ctx, &cloudformation.CreateChangeSetInput{
  213. ChangeSetName: aws.String(update),
  214. ChangeSetType: aws.String(cloudformation.ChangeSetTypeUpdate),
  215. StackName: aws.String(name),
  216. TemplateBody: aws.String(string(json)),
  217. Parameters: param,
  218. Capabilities: []*string{
  219. aws.String(cloudformation.CapabilityCapabilityIam),
  220. },
  221. })
  222. if err != nil {
  223. return "", err
  224. }
  225. err = s.CF.WaitUntilChangeSetCreateCompleteWithContext(ctx, &cloudformation.DescribeChangeSetInput{
  226. ChangeSetName: changeset.Id,
  227. })
  228. return *changeset.Id, err
  229. }
  230. func (s sdk) UpdateStack(ctx context.Context, changeset string) error {
  231. desc, err := s.CF.DescribeChangeSetWithContext(ctx, &cloudformation.DescribeChangeSetInput{
  232. ChangeSetName: aws.String(changeset),
  233. })
  234. if err != nil {
  235. return err
  236. }
  237. if strings.HasPrefix(aws.StringValue(desc.StatusReason), "The submitted information didn't contain changes.") {
  238. return nil
  239. }
  240. _, err = s.CF.ExecuteChangeSet(&cloudformation.ExecuteChangeSetInput{
  241. ChangeSetName: aws.String(changeset),
  242. })
  243. return err
  244. }
  245. const (
  246. stackCreate = iota
  247. stackUpdate
  248. stackDelete
  249. )
  250. func (s sdk) WaitStackComplete(ctx context.Context, name string, operation int) error {
  251. input := &cloudformation.DescribeStacksInput{
  252. StackName: aws.String(name),
  253. }
  254. switch operation {
  255. case stackCreate:
  256. return s.CF.WaitUntilStackCreateCompleteWithContext(ctx, input)
  257. case stackDelete:
  258. return s.CF.WaitUntilStackDeleteCompleteWithContext(ctx, input)
  259. default:
  260. return fmt.Errorf("internal error: unexpected stack operation %d", operation)
  261. }
  262. }
  263. func (s sdk) GetStackID(ctx context.Context, name string) (string, error) {
  264. stacks, err := s.CF.DescribeStacksWithContext(ctx, &cloudformation.DescribeStacksInput{
  265. StackName: aws.String(name),
  266. })
  267. if err != nil {
  268. return "", err
  269. }
  270. return *stacks.Stacks[0].StackId, nil
  271. }
  272. func (s sdk) ListStacks(ctx context.Context, name string) ([]compose.Stack, error) {
  273. cfStacks, err := s.CF.DescribeStacksWithContext(ctx, &cloudformation.DescribeStacksInput{})
  274. if err != nil {
  275. return nil, err
  276. }
  277. stacks := []compose.Stack{}
  278. for _, stack := range cfStacks.Stacks {
  279. for _, t := range stack.Tags {
  280. if *t.Key == compose.ProjectTag {
  281. status := compose.RUNNING
  282. switch aws.StringValue(stack.StackStatus) {
  283. case "CREATE_IN_PROGRESS":
  284. status = compose.STARTING
  285. case "DELETE_IN_PROGRESS":
  286. status = compose.REMOVING
  287. case "UPDATE_IN_PROGRESS":
  288. status = compose.UPDATING
  289. }
  290. stacks = append(stacks, compose.Stack{
  291. ID: aws.StringValue(stack.StackId),
  292. Name: aws.StringValue(stack.StackName),
  293. Status: status,
  294. })
  295. break
  296. }
  297. }
  298. }
  299. return stacks, nil
  300. }
  301. func (s sdk) DescribeStackEvents(ctx context.Context, stackID string) ([]*cloudformation.StackEvent, error) {
  302. // Fixme implement Paginator on Events and return as a chan(events)
  303. events := []*cloudformation.StackEvent{}
  304. var nextToken *string
  305. for {
  306. resp, err := s.CF.DescribeStackEventsWithContext(ctx, &cloudformation.DescribeStackEventsInput{
  307. StackName: aws.String(stackID),
  308. NextToken: nextToken,
  309. })
  310. if err != nil {
  311. return nil, err
  312. }
  313. events = append(events, resp.StackEvents...)
  314. if resp.NextToken == nil {
  315. return events, nil
  316. }
  317. nextToken = resp.NextToken
  318. }
  319. }
  320. func (s sdk) ListStackParameters(ctx context.Context, name string) (map[string]string, error) {
  321. st, err := s.CF.DescribeStacksWithContext(ctx, &cloudformation.DescribeStacksInput{
  322. NextToken: nil,
  323. StackName: aws.String(name),
  324. })
  325. if err != nil {
  326. return nil, err
  327. }
  328. parameters := map[string]string{}
  329. for _, parameter := range st.Stacks[0].Parameters {
  330. parameters[aws.StringValue(parameter.ParameterKey)] = aws.StringValue(parameter.ParameterValue)
  331. }
  332. return parameters, nil
  333. }
  334. type stackResource struct {
  335. LogicalID string
  336. Type string
  337. ARN string
  338. Status string
  339. }
  340. func (s sdk) ListStackResources(ctx context.Context, name string) ([]stackResource, error) {
  341. // FIXME handle pagination
  342. res, err := s.CF.ListStackResourcesWithContext(ctx, &cloudformation.ListStackResourcesInput{
  343. StackName: aws.String(name),
  344. })
  345. if err != nil {
  346. return nil, err
  347. }
  348. resources := []stackResource{}
  349. for _, r := range res.StackResourceSummaries {
  350. resources = append(resources, stackResource{
  351. LogicalID: aws.StringValue(r.LogicalResourceId),
  352. Type: aws.StringValue(r.ResourceType),
  353. ARN: aws.StringValue(r.PhysicalResourceId),
  354. Status: aws.StringValue(r.ResourceStatus),
  355. })
  356. }
  357. return resources, nil
  358. }
  359. func (s sdk) DeleteStack(ctx context.Context, name string) error {
  360. logrus.Debug("Delete CloudFormation stack")
  361. _, err := s.CF.DeleteStackWithContext(ctx, &cloudformation.DeleteStackInput{
  362. StackName: aws.String(name),
  363. })
  364. return err
  365. }
  366. func (s sdk) CreateSecret(ctx context.Context, secret secrets.Secret) (string, error) {
  367. logrus.Debug("Create secret " + secret.Name)
  368. secretStr, err := secret.GetCredString()
  369. if err != nil {
  370. return "", err
  371. }
  372. response, err := s.SM.CreateSecret(&secretsmanager.CreateSecretInput{
  373. Name: &secret.Name,
  374. SecretString: &secretStr,
  375. Description: &secret.Description,
  376. })
  377. if err != nil {
  378. return "", err
  379. }
  380. return aws.StringValue(response.ARN), nil
  381. }
  382. func (s sdk) InspectSecret(ctx context.Context, id string) (secrets.Secret, error) {
  383. logrus.Debug("Inspect secret " + id)
  384. response, err := s.SM.DescribeSecret(&secretsmanager.DescribeSecretInput{SecretId: &id})
  385. if err != nil {
  386. return secrets.Secret{}, err
  387. }
  388. labels := map[string]string{}
  389. for _, tag := range response.Tags {
  390. labels[aws.StringValue(tag.Key)] = aws.StringValue(tag.Value)
  391. }
  392. secret := secrets.Secret{
  393. ID: aws.StringValue(response.ARN),
  394. Name: aws.StringValue(response.Name),
  395. Labels: labels,
  396. }
  397. if response.Description != nil {
  398. secret.Description = *response.Description
  399. }
  400. return secret, nil
  401. }
  402. func (s sdk) ListSecrets(ctx context.Context) ([]secrets.Secret, error) {
  403. logrus.Debug("List secrets ...")
  404. response, err := s.SM.ListSecrets(&secretsmanager.ListSecretsInput{})
  405. if err != nil {
  406. return nil, err
  407. }
  408. var ls []secrets.Secret
  409. for _, sec := range response.SecretList {
  410. labels := map[string]string{}
  411. for _, tag := range sec.Tags {
  412. labels[*tag.Key] = *tag.Value
  413. }
  414. description := ""
  415. if sec.Description != nil {
  416. description = *sec.Description
  417. }
  418. ls = append(ls, secrets.Secret{
  419. ID: *sec.ARN,
  420. Name: *sec.Name,
  421. Labels: labels,
  422. Description: description,
  423. })
  424. }
  425. return ls, nil
  426. }
  427. func (s sdk) DeleteSecret(ctx context.Context, id string, recover bool) error {
  428. logrus.Debug("List secrets ...")
  429. force := !recover
  430. _, err := s.SM.DeleteSecret(&secretsmanager.DeleteSecretInput{SecretId: &id, ForceDeleteWithoutRecovery: &force})
  431. return err
  432. }
  433. func (s sdk) GetLogs(ctx context.Context, name string, consumer func(service, container, message string)) error {
  434. logGroup := fmt.Sprintf("/docker-compose/%s", name)
  435. var startTime int64
  436. for {
  437. select {
  438. case <-ctx.Done():
  439. return nil
  440. default:
  441. var hasMore = true
  442. var token *string
  443. for hasMore {
  444. events, err := s.CW.FilterLogEvents(&cloudwatchlogs.FilterLogEventsInput{
  445. LogGroupName: aws.String(logGroup),
  446. NextToken: token,
  447. StartTime: aws.Int64(startTime),
  448. })
  449. if err != nil {
  450. return err
  451. }
  452. if events.NextToken == nil {
  453. hasMore = false
  454. } else {
  455. token = events.NextToken
  456. }
  457. for _, event := range events.Events {
  458. p := strings.Split(aws.StringValue(event.LogStreamName), "/")
  459. consumer(p[1], p[2], aws.StringValue(event.Message))
  460. startTime = *event.IngestionTime
  461. }
  462. }
  463. }
  464. time.Sleep(500 * time.Millisecond)
  465. }
  466. }
  467. func (s sdk) DescribeServices(ctx context.Context, cluster string, arns []string) ([]compose.ServiceStatus, error) {
  468. services, err := s.ECS.DescribeServicesWithContext(ctx, &ecs.DescribeServicesInput{
  469. Cluster: aws.String(cluster),
  470. Services: aws.StringSlice(arns),
  471. Include: aws.StringSlice([]string{"TAGS"}),
  472. })
  473. if err != nil {
  474. return nil, err
  475. }
  476. status := []compose.ServiceStatus{}
  477. for _, service := range services.Services {
  478. var name string
  479. for _, t := range service.Tags {
  480. if *t.Key == compose.ServiceTag {
  481. name = aws.StringValue(t.Value)
  482. }
  483. }
  484. if name == "" {
  485. return nil, fmt.Errorf("service %s doesn't have a %s tag", *service.ServiceArn, compose.ServiceTag)
  486. }
  487. targetGroupArns := []string{}
  488. for _, lb := range service.LoadBalancers {
  489. targetGroupArns = append(targetGroupArns, *lb.TargetGroupArn)
  490. }
  491. // getURLwithPortMapping makes 2 queries
  492. // one to get the target groups and another for load balancers
  493. loadBalancers, err := s.getURLWithPortMapping(ctx, targetGroupArns)
  494. if err != nil {
  495. return nil, err
  496. }
  497. status = append(status, compose.ServiceStatus{
  498. ID: aws.StringValue(service.ServiceName),
  499. Name: name,
  500. Replicas: int(aws.Int64Value(service.RunningCount)),
  501. Desired: int(aws.Int64Value(service.DesiredCount)),
  502. Publishers: loadBalancers,
  503. })
  504. }
  505. return status, nil
  506. }
  507. func (s sdk) getURLWithPortMapping(ctx context.Context, targetGroupArns []string) ([]compose.PortPublisher, error) {
  508. if len(targetGroupArns) == 0 {
  509. return nil, nil
  510. }
  511. groups, err := s.ELB.DescribeTargetGroups(&elbv2.DescribeTargetGroupsInput{
  512. TargetGroupArns: aws.StringSlice(targetGroupArns),
  513. })
  514. if err != nil {
  515. return nil, err
  516. }
  517. lbarns := []*string{}
  518. for _, tg := range groups.TargetGroups {
  519. lbarns = append(lbarns, tg.LoadBalancerArns...)
  520. }
  521. lbs, err := s.ELB.DescribeLoadBalancersWithContext(ctx, &elbv2.DescribeLoadBalancersInput{
  522. LoadBalancerArns: lbarns,
  523. })
  524. if err != nil {
  525. return nil, err
  526. }
  527. filterLB := func(arn *string, lbs []*elbv2.LoadBalancer) *elbv2.LoadBalancer {
  528. if aws.StringValue(arn) == "" {
  529. // load balancer arn is nil/""
  530. return nil
  531. }
  532. for _, lb := range lbs {
  533. if aws.StringValue(lb.LoadBalancerArn) == aws.StringValue(arn) {
  534. return lb
  535. }
  536. }
  537. return nil
  538. }
  539. loadBalancers := []compose.PortPublisher{}
  540. for _, tg := range groups.TargetGroups {
  541. for _, lbarn := range tg.LoadBalancerArns {
  542. lb := filterLB(lbarn, lbs.LoadBalancers)
  543. if lb == nil {
  544. continue
  545. }
  546. loadBalancers = append(loadBalancers, compose.PortPublisher{
  547. URL: aws.StringValue(lb.DNSName),
  548. TargetPort: int(aws.Int64Value(tg.Port)),
  549. PublishedPort: int(aws.Int64Value(tg.Port)),
  550. Protocol: aws.StringValue(tg.Protocol),
  551. })
  552. }
  553. }
  554. return loadBalancers, nil
  555. }
  556. func (s sdk) ListTasks(ctx context.Context, cluster string, family string) ([]string, error) {
  557. tasks, err := s.ECS.ListTasksWithContext(ctx, &ecs.ListTasksInput{
  558. Cluster: aws.String(cluster),
  559. Family: aws.String(family),
  560. })
  561. if err != nil {
  562. return nil, err
  563. }
  564. arns := []string{}
  565. for _, arn := range tasks.TaskArns {
  566. arns = append(arns, *arn)
  567. }
  568. return arns, nil
  569. }
  570. func (s sdk) GetPublicIPs(ctx context.Context, interfaces ...string) (map[string]string, error) {
  571. desc, err := s.EC2.DescribeNetworkInterfaces(&ec2.DescribeNetworkInterfacesInput{
  572. NetworkInterfaceIds: aws.StringSlice(interfaces),
  573. })
  574. if err != nil {
  575. return nil, err
  576. }
  577. publicIPs := map[string]string{}
  578. for _, interf := range desc.NetworkInterfaces {
  579. if interf.Association != nil {
  580. publicIPs[aws.StringValue(interf.NetworkInterfaceId)] = aws.StringValue(interf.Association.PublicIp)
  581. }
  582. }
  583. return publicIPs, nil
  584. }
  585. func (s sdk) LoadBalancerExists(ctx context.Context, arn string) (bool, error) {
  586. logrus.Debug("CheckRequirements if PortPublisher exists: ", arn)
  587. lbs, err := s.ELB.DescribeLoadBalancersWithContext(ctx, &elbv2.DescribeLoadBalancersInput{
  588. LoadBalancerArns: []*string{aws.String(arn)},
  589. })
  590. if err != nil {
  591. return false, err
  592. }
  593. return len(lbs.LoadBalancers) > 0, nil
  594. }
  595. func (s sdk) GetLoadBalancerURL(ctx context.Context, arn string) (string, error) {
  596. logrus.Debug("Retrieve load balancer URL: ", arn)
  597. lbs, err := s.ELB.DescribeLoadBalancersWithContext(ctx, &elbv2.DescribeLoadBalancersInput{
  598. LoadBalancerArns: []*string{aws.String(arn)},
  599. })
  600. if err != nil {
  601. return "", err
  602. }
  603. dnsName := aws.StringValue(lbs.LoadBalancers[0].DNSName)
  604. if dnsName == "" {
  605. return "", fmt.Errorf("Load balancer %s doesn't have a dns name", aws.StringValue(lbs.LoadBalancers[0].LoadBalancerArn))
  606. }
  607. return dnsName, nil
  608. }