1
0

sdk.go 35 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246
  1. /*
  2. Copyright 2020 Docker Compose CLI authors
  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. "bytes"
  16. "context"
  17. "encoding/json"
  18. "fmt"
  19. "strings"
  20. "time"
  21. "github.com/docker/compose-cli/api/compose"
  22. "github.com/docker/compose-cli/api/errdefs"
  23. "github.com/docker/compose-cli/api/secrets"
  24. "github.com/docker/compose-cli/internal"
  25. "github.com/aws/aws-sdk-go/aws"
  26. "github.com/aws/aws-sdk-go/aws/arn"
  27. "github.com/aws/aws-sdk-go/aws/request"
  28. "github.com/aws/aws-sdk-go/aws/session"
  29. "github.com/aws/aws-sdk-go/service/autoscaling"
  30. "github.com/aws/aws-sdk-go/service/autoscaling/autoscalingiface"
  31. "github.com/aws/aws-sdk-go/service/cloudformation"
  32. "github.com/aws/aws-sdk-go/service/cloudformation/cloudformationiface"
  33. "github.com/aws/aws-sdk-go/service/cloudwatchlogs"
  34. "github.com/aws/aws-sdk-go/service/cloudwatchlogs/cloudwatchlogsiface"
  35. "github.com/aws/aws-sdk-go/service/ec2"
  36. "github.com/aws/aws-sdk-go/service/ec2/ec2iface"
  37. "github.com/aws/aws-sdk-go/service/ecs"
  38. "github.com/aws/aws-sdk-go/service/ecs/ecsiface"
  39. "github.com/aws/aws-sdk-go/service/efs"
  40. "github.com/aws/aws-sdk-go/service/efs/efsiface"
  41. "github.com/aws/aws-sdk-go/service/elbv2"
  42. "github.com/aws/aws-sdk-go/service/elbv2/elbv2iface"
  43. "github.com/aws/aws-sdk-go/service/iam"
  44. "github.com/aws/aws-sdk-go/service/iam/iamiface"
  45. "github.com/aws/aws-sdk-go/service/s3"
  46. "github.com/aws/aws-sdk-go/service/s3/s3iface"
  47. "github.com/aws/aws-sdk-go/service/s3/s3manager"
  48. "github.com/aws/aws-sdk-go/service/secretsmanager"
  49. "github.com/aws/aws-sdk-go/service/secretsmanager/secretsmanageriface"
  50. "github.com/aws/aws-sdk-go/service/ssm"
  51. "github.com/aws/aws-sdk-go/service/ssm/ssmiface"
  52. "github.com/hashicorp/go-multierror"
  53. "github.com/hashicorp/go-uuid"
  54. "github.com/pkg/errors"
  55. "github.com/sirupsen/logrus"
  56. )
  57. type sdk struct {
  58. ECS ecsiface.ECSAPI
  59. EC2 ec2iface.EC2API
  60. EFS efsiface.EFSAPI
  61. ELB elbv2iface.ELBV2API
  62. CW cloudwatchlogsiface.CloudWatchLogsAPI
  63. IAM iamiface.IAMAPI
  64. CF cloudformationiface.CloudFormationAPI
  65. SM secretsmanageriface.SecretsManagerAPI
  66. SSM ssmiface.SSMAPI
  67. AG autoscalingiface.AutoScalingAPI
  68. S3 s3iface.S3API
  69. uploader *s3manager.Uploader
  70. }
  71. // sdk implement API
  72. var _ API = sdk{}
  73. func newSDK(sess *session.Session) sdk {
  74. sess.Handlers.Build.PushBack(func(r *request.Request) {
  75. request.AddToUserAgent(r, internal.ECSUserAgentName+"/"+internal.Version)
  76. })
  77. return sdk{
  78. ECS: ecs.New(sess),
  79. EC2: ec2.New(sess),
  80. EFS: efs.New(sess),
  81. ELB: elbv2.New(sess),
  82. CW: cloudwatchlogs.New(sess),
  83. IAM: iam.New(sess),
  84. CF: cloudformation.New(sess),
  85. SM: secretsmanager.New(sess),
  86. SSM: ssm.New(sess),
  87. AG: autoscaling.New(sess),
  88. S3: s3.New(sess),
  89. uploader: s3manager.NewUploader(sess),
  90. }
  91. }
  92. func (s sdk) CheckRequirements(ctx context.Context, region string) error {
  93. settings, err := s.ECS.ListAccountSettingsWithContext(ctx, &ecs.ListAccountSettingsInput{
  94. EffectiveSettings: aws.Bool(true),
  95. Name: aws.String("serviceLongArnFormat"),
  96. })
  97. if err != nil {
  98. return err
  99. }
  100. serviceLongArnFormat := settings.Settings[0].Value
  101. if *serviceLongArnFormat != "enabled" {
  102. return fmt.Errorf("this tool requires the \"new ARN resource ID format\".\n"+
  103. "Check https://%s.console.aws.amazon.com/ecs/home#/settings\n"+
  104. "Learn more: https://aws.amazon.com/blogs/compute/migrating-your-amazon-ecs-deployment-to-the-new-arn-and-resource-id-format-2", region)
  105. }
  106. return nil
  107. }
  108. func (s sdk) ResolveCluster(ctx context.Context, nameOrArn string) (awsResource, error) {
  109. logrus.Debug("CheckRequirements if cluster was already created: ", nameOrArn)
  110. clusters, err := s.ECS.DescribeClustersWithContext(ctx, &ecs.DescribeClustersInput{
  111. Clusters: []*string{aws.String(nameOrArn)},
  112. })
  113. if err != nil {
  114. return nil, err
  115. }
  116. if len(clusters.Clusters) == 0 {
  117. return nil, errors.Wrapf(errdefs.ErrNotFound, "cluster %q does not exist", nameOrArn)
  118. }
  119. it := clusters.Clusters[0]
  120. return existingAWSResource{
  121. arn: aws.StringValue(it.ClusterArn),
  122. id: aws.StringValue(it.ClusterName),
  123. }, nil
  124. }
  125. func (s sdk) CreateCluster(ctx context.Context, name string) (string, error) {
  126. logrus.Debug("Create cluster ", name)
  127. response, err := s.ECS.CreateClusterWithContext(ctx, &ecs.CreateClusterInput{ClusterName: aws.String(name)})
  128. if err != nil {
  129. return "", err
  130. }
  131. return *response.Cluster.Status, nil
  132. }
  133. func (s sdk) CheckVPC(ctx context.Context, vpcID string) error {
  134. logrus.Debug("CheckRequirements on VPC : ", vpcID)
  135. output, err := s.EC2.DescribeVpcAttributeWithContext(ctx, &ec2.DescribeVpcAttributeInput{
  136. VpcId: aws.String(vpcID),
  137. Attribute: aws.String("enableDnsSupport"),
  138. })
  139. if err != nil {
  140. return err
  141. }
  142. if !*output.EnableDnsSupport.Value {
  143. return fmt.Errorf("VPC %q doesn't have DNS resolution enabled", vpcID)
  144. }
  145. return nil
  146. }
  147. func (s sdk) GetDefaultVPC(ctx context.Context) (string, error) {
  148. logrus.Debug("Retrieve default VPC")
  149. vpcs, err := s.EC2.DescribeVpcsWithContext(ctx, &ec2.DescribeVpcsInput{
  150. Filters: []*ec2.Filter{
  151. {
  152. Name: aws.String("isDefault"),
  153. Values: []*string{aws.String("true")},
  154. },
  155. },
  156. })
  157. if err != nil {
  158. return "", err
  159. }
  160. if len(vpcs.Vpcs) == 0 {
  161. return "", fmt.Errorf("account has not default VPC")
  162. }
  163. return *vpcs.Vpcs[0].VpcId, nil
  164. }
  165. func (s sdk) GetSubNets(ctx context.Context, vpcID string) ([]awsResource, error) {
  166. logrus.Debug("Retrieve SubNets")
  167. var ids []awsResource
  168. var token *string
  169. for {
  170. subnets, err := s.EC2.DescribeSubnetsWithContext(ctx, &ec2.DescribeSubnetsInput{
  171. Filters: []*ec2.Filter{
  172. {
  173. Name: aws.String("vpc-id"),
  174. Values: []*string{aws.String(vpcID)},
  175. },
  176. },
  177. NextToken: token,
  178. })
  179. if err != nil {
  180. return nil, err
  181. }
  182. for _, subnet := range subnets.Subnets {
  183. ids = append(ids, existingAWSResource{
  184. arn: aws.StringValue(subnet.SubnetArn),
  185. id: aws.StringValue(subnet.SubnetId),
  186. })
  187. }
  188. if subnets.NextToken == token {
  189. break
  190. }
  191. token = subnets.NextToken
  192. }
  193. return ids, nil
  194. }
  195. func (s sdk) IsPublicSubnet(ctx context.Context, subNetID string) (bool, error) {
  196. tables, err := s.EC2.DescribeRouteTablesWithContext(ctx, &ec2.DescribeRouteTablesInput{
  197. Filters: []*ec2.Filter{
  198. {
  199. Name: aws.String("association.subnet-id"),
  200. Values: []*string{aws.String(subNetID)},
  201. },
  202. },
  203. })
  204. if err != nil {
  205. return false, err
  206. }
  207. if len(tables.RouteTables) == 0 {
  208. // If a subnet is not explicitly associated with any route table, it is implicitly associated with the main route table.
  209. // https://docs.aws.amazon.com/cli/latest/reference/ec2/describe-route-tables.html
  210. return true, nil
  211. }
  212. for _, routeTable := range tables.RouteTables {
  213. for _, route := range routeTable.Routes {
  214. if aws.StringValue(route.State) != "active" {
  215. continue
  216. }
  217. if strings.HasPrefix(aws.StringValue(route.GatewayId), "igw-") {
  218. // Connected to an internet Gateway
  219. return true, nil
  220. }
  221. }
  222. }
  223. return false, nil
  224. }
  225. func (s sdk) GetRoleArn(ctx context.Context, name string) (string, error) {
  226. role, err := s.IAM.GetRoleWithContext(ctx, &iam.GetRoleInput{
  227. RoleName: aws.String(name),
  228. })
  229. if err != nil {
  230. return "", err
  231. }
  232. return *role.Role.Arn, nil
  233. }
  234. func (s sdk) StackExists(ctx context.Context, name string) (bool, error) {
  235. stacks, err := s.CF.DescribeStacksWithContext(ctx, &cloudformation.DescribeStacksInput{
  236. StackName: aws.String(name),
  237. })
  238. if err != nil {
  239. if strings.HasPrefix(err.Error(), fmt.Sprintf("ValidationError: Stack with ID %s does not exist", name)) {
  240. return false, nil
  241. }
  242. return false, nil
  243. }
  244. return len(stacks.Stacks) > 0, nil
  245. }
  246. type uploadedTemplateFunc func(body *string, url *string) (string, error)
  247. const cloudformationBytesLimit = 51200
  248. func (s sdk) withTemplate(ctx context.Context, name string, template []byte, region string, fn uploadedTemplateFunc) (string, error) {
  249. if len(template) < cloudformationBytesLimit {
  250. return fn(aws.String(string(template)), nil)
  251. }
  252. key, err := uuid.GenerateUUID()
  253. if err != nil {
  254. return "", err
  255. }
  256. bucket := "com.docker.compose." + key
  257. logrus.Debugf("Create s3 bucket %q to store cloudformation template", bucket)
  258. var configuration *s3.CreateBucketConfiguration
  259. if region != "us-east-1" {
  260. configuration = &s3.CreateBucketConfiguration{
  261. LocationConstraint: aws.String(region),
  262. }
  263. }
  264. _, err = s.S3.CreateBucket(&s3.CreateBucketInput{
  265. Bucket: aws.String(bucket),
  266. CreateBucketConfiguration: configuration,
  267. })
  268. if err != nil {
  269. return "", err
  270. }
  271. upload, err := s.uploader.UploadWithContext(ctx, &s3manager.UploadInput{
  272. Key: aws.String("template.yaml"),
  273. Body: bytes.NewReader(template),
  274. Bucket: aws.String(bucket),
  275. ContentType: aws.String("application/x-yaml"),
  276. Tagging: aws.String(name),
  277. })
  278. if err != nil {
  279. return "", err
  280. }
  281. defer func() {
  282. _, err := s.S3.DeleteObjectWithContext(ctx, &s3.DeleteObjectInput{
  283. Bucket: aws.String(bucket),
  284. Key: aws.String("template.yaml"),
  285. VersionId: upload.VersionID,
  286. })
  287. if err != nil {
  288. logrus.Warnf("Failed to remove S3 bucket: %s", err)
  289. }
  290. _, err = s.S3.DeleteBucketWithContext(ctx, &s3.DeleteBucketInput{
  291. Bucket: aws.String(bucket),
  292. })
  293. if err != nil {
  294. logrus.Warnf("Failed to remove S3 bucket: %s", err)
  295. }
  296. }()
  297. return fn(nil, aws.String(upload.Location))
  298. }
  299. func (s sdk) CreateStack(ctx context.Context, name string, region string, template []byte) error {
  300. logrus.Debug("Create CloudFormation stack")
  301. stackID, err := s.withTemplate(ctx, name, template, region, func(body *string, url *string) (string, error) {
  302. stack, err := s.CF.CreateStackWithContext(ctx, &cloudformation.CreateStackInput{
  303. OnFailure: aws.String("DELETE"),
  304. StackName: aws.String(name),
  305. TemplateBody: body,
  306. TemplateURL: url,
  307. TimeoutInMinutes: nil,
  308. Capabilities: []*string{
  309. aws.String(cloudformation.CapabilityCapabilityIam),
  310. },
  311. Tags: []*cloudformation.Tag{
  312. {
  313. Key: aws.String(compose.ProjectTag),
  314. Value: aws.String(name),
  315. },
  316. },
  317. })
  318. if err != nil {
  319. return "", err
  320. }
  321. return aws.StringValue(stack.StackId), nil
  322. })
  323. logrus.Debugf("Stack %s created", stackID)
  324. return err
  325. }
  326. func (s sdk) CreateChangeSet(ctx context.Context, name string, region string, template []byte) (string, error) {
  327. logrus.Debug("Create CloudFormation Changeset")
  328. update := fmt.Sprintf("Update%s", time.Now().Format("2006-01-02-15-04-05"))
  329. changeset, err := s.withTemplate(ctx, name, template, region, func(body *string, url *string) (string, error) {
  330. changeset, err := s.CF.CreateChangeSetWithContext(ctx, &cloudformation.CreateChangeSetInput{
  331. ChangeSetName: aws.String(update),
  332. ChangeSetType: aws.String(cloudformation.ChangeSetTypeUpdate),
  333. StackName: aws.String(name),
  334. TemplateBody: body,
  335. TemplateURL: url,
  336. Capabilities: []*string{
  337. aws.String(cloudformation.CapabilityCapabilityIam),
  338. },
  339. })
  340. if err != nil {
  341. return "", err
  342. }
  343. return aws.StringValue(changeset.Id), err
  344. })
  345. if err != nil {
  346. return "", err
  347. }
  348. // we have to WaitUntilChangeSetCreateComplete even this in fail with error `ResourceNotReady`
  349. // so that we can invoke DescribeChangeSet to check status, and then we can know about the actual creation failure cause.
  350. s.CF.WaitUntilChangeSetCreateCompleteWithContext(ctx, &cloudformation.DescribeChangeSetInput{ // nolint:errcheck
  351. ChangeSetName: aws.String(changeset),
  352. })
  353. desc, err := s.CF.DescribeChangeSetWithContext(ctx, &cloudformation.DescribeChangeSetInput{
  354. ChangeSetName: aws.String(update),
  355. StackName: aws.String(name),
  356. })
  357. if aws.StringValue(desc.Status) == "FAILED" {
  358. return changeset, fmt.Errorf(aws.StringValue(desc.StatusReason))
  359. }
  360. return changeset, err
  361. }
  362. func (s sdk) UpdateStack(ctx context.Context, changeset string) error {
  363. desc, err := s.CF.DescribeChangeSetWithContext(ctx, &cloudformation.DescribeChangeSetInput{
  364. ChangeSetName: aws.String(changeset),
  365. })
  366. if err != nil {
  367. return err
  368. }
  369. if strings.HasPrefix(aws.StringValue(desc.StatusReason), "The submitted information didn't contain changes.") {
  370. return nil
  371. }
  372. _, err = s.CF.ExecuteChangeSet(&cloudformation.ExecuteChangeSetInput{
  373. ChangeSetName: aws.String(changeset),
  374. })
  375. return err
  376. }
  377. const (
  378. stackCreate = iota
  379. stackUpdate
  380. stackDelete
  381. )
  382. func (s sdk) WaitStackComplete(ctx context.Context, name string, operation int) error {
  383. input := &cloudformation.DescribeStacksInput{
  384. StackName: aws.String(name),
  385. }
  386. switch operation {
  387. case stackCreate:
  388. return s.CF.WaitUntilStackCreateCompleteWithContext(ctx, input)
  389. case stackDelete:
  390. return s.CF.WaitUntilStackDeleteCompleteWithContext(ctx, input)
  391. default:
  392. return fmt.Errorf("internal error: unexpected stack operation %d", operation)
  393. }
  394. }
  395. func (s sdk) GetStackID(ctx context.Context, name string) (string, error) {
  396. stacks, err := s.CF.DescribeStacksWithContext(ctx, &cloudformation.DescribeStacksInput{
  397. StackName: aws.String(name),
  398. })
  399. if err != nil {
  400. return "", err
  401. }
  402. return *stacks.Stacks[0].StackId, nil
  403. }
  404. func (s sdk) ListStacks(ctx context.Context) ([]compose.Stack, error) {
  405. params := cloudformation.DescribeStacksInput{}
  406. var token *string
  407. var stacks []compose.Stack
  408. for {
  409. response, err := s.CF.DescribeStacksWithContext(ctx, &params)
  410. if err != nil {
  411. return nil, err
  412. }
  413. for _, stack := range response.Stacks {
  414. for _, t := range stack.Tags {
  415. if *t.Key == compose.ProjectTag {
  416. status := compose.RUNNING
  417. switch aws.StringValue(stack.StackStatus) {
  418. case "CREATE_IN_PROGRESS":
  419. status = compose.STARTING
  420. case "DELETE_IN_PROGRESS":
  421. status = compose.REMOVING
  422. case "UPDATE_IN_PROGRESS":
  423. status = compose.UPDATING
  424. default:
  425. }
  426. stacks = append(stacks, compose.Stack{
  427. ID: aws.StringValue(stack.StackId),
  428. Name: aws.StringValue(stack.StackName),
  429. Status: status,
  430. })
  431. break
  432. }
  433. }
  434. }
  435. if token == response.NextToken {
  436. return stacks, nil
  437. }
  438. token = response.NextToken
  439. }
  440. }
  441. func (s sdk) GetStackClusterID(ctx context.Context, stack string) (string, error) {
  442. // Note: could use DescribeStackResource but we only can detect `does not exist` case by matching string error message
  443. var token *string
  444. for {
  445. response, err := s.CF.ListStackResourcesWithContext(ctx, &cloudformation.ListStackResourcesInput{
  446. StackName: aws.String(stack),
  447. })
  448. if err != nil {
  449. return "", err
  450. }
  451. for _, r := range response.StackResourceSummaries {
  452. if aws.StringValue(r.ResourceType) == "AWS::ECS::Cluster" {
  453. return aws.StringValue(r.PhysicalResourceId), nil
  454. }
  455. }
  456. if token == response.NextToken {
  457. break
  458. }
  459. token = response.NextToken
  460. }
  461. // stack is using user-provided cluster
  462. res, err := s.CF.GetTemplateSummaryWithContext(ctx, &cloudformation.GetTemplateSummaryInput{
  463. StackName: aws.String(stack),
  464. })
  465. if err != nil {
  466. return "", err
  467. }
  468. c := aws.StringValue(res.Metadata)
  469. var m templateMetadata
  470. err = json.Unmarshal([]byte(c), &m)
  471. if err != nil {
  472. return "", err
  473. }
  474. if m.Cluster == "" {
  475. return "", errors.Wrap(errdefs.ErrNotFound, "CloudFormation is missing cluster metadata")
  476. }
  477. return m.Cluster, nil
  478. }
  479. type templateMetadata struct {
  480. Cluster string `json:",omitempty"`
  481. }
  482. func (s sdk) GetServiceTaskDefinition(ctx context.Context, cluster string, serviceArns []string) (map[string]string, error) {
  483. defs := map[string]string{}
  484. svc := []*string{}
  485. for _, s := range serviceArns {
  486. svc = append(svc, aws.String(s))
  487. }
  488. for i := 0; i < len(svc); i += 10 {
  489. end := i + 10
  490. if end > len(svc) {
  491. end = len(svc)
  492. }
  493. chunk := svc[i:end]
  494. services, err := s.ECS.DescribeServicesWithContext(ctx, &ecs.DescribeServicesInput{
  495. Cluster: aws.String(cluster),
  496. Services: chunk,
  497. })
  498. if err != nil {
  499. return nil, err
  500. }
  501. for _, s := range services.Services {
  502. defs[aws.StringValue(s.ServiceArn)] = aws.StringValue(s.TaskDefinition)
  503. }
  504. }
  505. return defs, nil
  506. }
  507. func (s sdk) ListStackServices(ctx context.Context, stack string) ([]string, error) {
  508. arns := []string{}
  509. var nextToken *string
  510. for {
  511. response, err := s.CF.ListStackResourcesWithContext(ctx, &cloudformation.ListStackResourcesInput{
  512. StackName: aws.String(stack),
  513. NextToken: nextToken,
  514. })
  515. if err != nil {
  516. return nil, err
  517. }
  518. for _, r := range response.StackResourceSummaries {
  519. if aws.StringValue(r.ResourceType) == "AWS::ECS::Service" {
  520. if r.PhysicalResourceId != nil {
  521. arns = append(arns, aws.StringValue(r.PhysicalResourceId))
  522. }
  523. }
  524. }
  525. nextToken = response.NextToken
  526. if nextToken == nil {
  527. break
  528. }
  529. }
  530. return arns, nil
  531. }
  532. func (s sdk) GetServiceTasks(ctx context.Context, cluster string, service string, stopped bool) ([]*ecs.Task, error) {
  533. state := "RUNNING"
  534. if stopped {
  535. state = "STOPPED"
  536. }
  537. var token *string
  538. var tasks []*ecs.Task
  539. for {
  540. response, err := s.ECS.ListTasksWithContext(ctx, &ecs.ListTasksInput{
  541. Cluster: aws.String(cluster),
  542. ServiceName: aws.String(service),
  543. DesiredStatus: aws.String(state),
  544. })
  545. if err != nil {
  546. return nil, err
  547. }
  548. if len(response.TaskArns) > 0 {
  549. taskDescriptions, err := s.ECS.DescribeTasksWithContext(ctx, &ecs.DescribeTasksInput{
  550. Cluster: aws.String(cluster),
  551. Tasks: response.TaskArns,
  552. })
  553. if err != nil {
  554. return nil, err
  555. }
  556. tasks = append(tasks, taskDescriptions.Tasks...)
  557. }
  558. if token == response.NextToken {
  559. return tasks, nil
  560. }
  561. token = response.NextToken
  562. }
  563. }
  564. func (s sdk) GetTaskStoppedReason(ctx context.Context, cluster string, taskArn string) (string, error) {
  565. taskDescriptions, err := s.ECS.DescribeTasksWithContext(ctx, &ecs.DescribeTasksInput{
  566. Cluster: aws.String(cluster),
  567. Tasks: []*string{aws.String(taskArn)},
  568. })
  569. if err != nil {
  570. return "", err
  571. }
  572. if len(taskDescriptions.Tasks) == 0 {
  573. return "", nil
  574. }
  575. task := taskDescriptions.Tasks[0]
  576. return fmt.Sprintf(
  577. "%s: %s",
  578. aws.StringValue(task.StopCode),
  579. aws.StringValue(task.StoppedReason)), nil
  580. }
  581. func (s sdk) DescribeStackEvents(ctx context.Context, stackID string) ([]*cloudformation.StackEvent, error) {
  582. // Fixme implement Paginator on Events and return as a chan(events)
  583. events := []*cloudformation.StackEvent{}
  584. var nextToken *string
  585. for {
  586. resp, err := s.CF.DescribeStackEventsWithContext(ctx, &cloudformation.DescribeStackEventsInput{
  587. StackName: aws.String(stackID),
  588. NextToken: nextToken,
  589. })
  590. if err != nil {
  591. return nil, err
  592. }
  593. events = append(events, resp.StackEvents...)
  594. if resp.NextToken == nil {
  595. return events, nil
  596. }
  597. nextToken = resp.NextToken
  598. }
  599. }
  600. func (s sdk) ListStackParameters(ctx context.Context, name string) (map[string]string, error) {
  601. st, err := s.CF.DescribeStacksWithContext(ctx, &cloudformation.DescribeStacksInput{
  602. NextToken: nil,
  603. StackName: aws.String(name),
  604. })
  605. if err != nil {
  606. return nil, err
  607. }
  608. parameters := map[string]string{}
  609. for _, parameter := range st.Stacks[0].Parameters {
  610. parameters[aws.StringValue(parameter.ParameterKey)] = aws.StringValue(parameter.ParameterValue)
  611. }
  612. return parameters, nil
  613. }
  614. type stackResource struct {
  615. LogicalID string
  616. Type string
  617. ARN string
  618. Status string
  619. }
  620. type stackResourceFn func(r stackResource) error
  621. type stackResources []stackResource
  622. func (resources stackResources) apply(awsType string, fn stackResourceFn) error {
  623. var errs *multierror.Error
  624. for _, r := range resources {
  625. if r.Type == awsType {
  626. err := fn(r)
  627. if err != nil {
  628. errs = multierror.Append(err)
  629. }
  630. }
  631. }
  632. return errs.ErrorOrNil()
  633. }
  634. func (s sdk) ListStackResources(ctx context.Context, name string) (stackResources, error) {
  635. var token *string
  636. var resources stackResources
  637. for {
  638. response, err := s.CF.ListStackResourcesWithContext(ctx, &cloudformation.ListStackResourcesInput{
  639. StackName: aws.String(name),
  640. })
  641. if err != nil {
  642. return nil, err
  643. }
  644. for _, r := range response.StackResourceSummaries {
  645. resources = append(resources, stackResource{
  646. LogicalID: aws.StringValue(r.LogicalResourceId),
  647. Type: aws.StringValue(r.ResourceType),
  648. ARN: aws.StringValue(r.PhysicalResourceId),
  649. Status: aws.StringValue(r.ResourceStatus),
  650. })
  651. }
  652. if token == response.NextToken {
  653. return resources, nil
  654. }
  655. token = response.NextToken
  656. }
  657. }
  658. func (s sdk) DeleteStack(ctx context.Context, name string) error {
  659. logrus.Debug("Delete CloudFormation stack")
  660. _, err := s.CF.DeleteStackWithContext(ctx, &cloudformation.DeleteStackInput{
  661. StackName: aws.String(name),
  662. })
  663. return err
  664. }
  665. func (s sdk) CreateSecret(ctx context.Context, secret secrets.Secret) (string, error) {
  666. logrus.Debug("Create secret " + secret.Name)
  667. var tags []*secretsmanager.Tag
  668. for k, v := range secret.Labels {
  669. tags = []*secretsmanager.Tag{
  670. {
  671. Key: aws.String(k),
  672. Value: aws.String(v),
  673. },
  674. }
  675. }
  676. // store the secret content as string
  677. content := string(secret.GetContent())
  678. response, err := s.SM.CreateSecret(&secretsmanager.CreateSecretInput{
  679. Name: &secret.Name,
  680. SecretString: &content,
  681. Tags: tags,
  682. })
  683. if err != nil {
  684. return "", err
  685. }
  686. return aws.StringValue(response.ARN), nil
  687. }
  688. func (s sdk) InspectSecret(ctx context.Context, id string) (secrets.Secret, error) {
  689. logrus.Debug("Inspect secret " + id)
  690. response, err := s.SM.DescribeSecret(&secretsmanager.DescribeSecretInput{SecretId: &id})
  691. if err != nil {
  692. return secrets.Secret{}, err
  693. }
  694. tags := map[string]string{}
  695. for _, tag := range response.Tags {
  696. tags[aws.StringValue(tag.Key)] = aws.StringValue(tag.Value)
  697. }
  698. secret := secrets.Secret{
  699. ID: aws.StringValue(response.ARN),
  700. Name: aws.StringValue(response.Name),
  701. Labels: tags,
  702. }
  703. return secret, nil
  704. }
  705. func (s sdk) ListSecrets(ctx context.Context) ([]secrets.Secret, error) {
  706. logrus.Debug("List secrets ...")
  707. var ls []secrets.Secret
  708. var token *string
  709. for {
  710. response, err := s.SM.ListSecrets(&secretsmanager.ListSecretsInput{})
  711. if err != nil {
  712. return nil, err
  713. }
  714. for _, sec := range response.SecretList {
  715. tags := map[string]string{}
  716. for _, tag := range sec.Tags {
  717. tags[*tag.Key] = *tag.Value
  718. }
  719. ls = append(ls, secrets.Secret{
  720. ID: *sec.ARN,
  721. Name: *sec.Name,
  722. Labels: tags,
  723. })
  724. }
  725. if token == response.NextToken {
  726. return ls, nil
  727. }
  728. token = response.NextToken
  729. }
  730. }
  731. func (s sdk) DeleteSecret(ctx context.Context, id string, recover bool) error {
  732. logrus.Debug("List secrets ...")
  733. force := !recover
  734. _, err := s.SM.DeleteSecret(&secretsmanager.DeleteSecretInput{SecretId: &id, ForceDeleteWithoutRecovery: &force})
  735. return err
  736. }
  737. func (s sdk) GetLogs(ctx context.Context, name string, consumer func(service, container, message string)) error {
  738. logGroup := fmt.Sprintf("/docker-compose/%s", name)
  739. var startTime int64
  740. for {
  741. select {
  742. case <-ctx.Done():
  743. return nil
  744. default:
  745. var hasMore = true
  746. var token *string
  747. for hasMore {
  748. events, err := s.CW.FilterLogEvents(&cloudwatchlogs.FilterLogEventsInput{
  749. LogGroupName: aws.String(logGroup),
  750. NextToken: token,
  751. StartTime: aws.Int64(startTime),
  752. })
  753. if err != nil {
  754. return err
  755. }
  756. if events.NextToken == nil {
  757. hasMore = false
  758. } else {
  759. token = events.NextToken
  760. }
  761. for _, event := range events.Events {
  762. p := strings.Split(aws.StringValue(event.LogStreamName), "/")
  763. consumer(p[1], p[2], aws.StringValue(event.Message))
  764. startTime = *event.IngestionTime
  765. }
  766. }
  767. }
  768. time.Sleep(500 * time.Millisecond)
  769. }
  770. }
  771. func (s sdk) DescribeService(ctx context.Context, cluster string, arn string) (compose.ServiceStatus, error) {
  772. services, err := s.ECS.DescribeServicesWithContext(ctx, &ecs.DescribeServicesInput{
  773. Cluster: aws.String(cluster),
  774. Services: []*string{aws.String(arn)},
  775. Include: aws.StringSlice([]string{"TAGS"}),
  776. })
  777. if err != nil {
  778. return compose.ServiceStatus{}, err
  779. }
  780. for _, f := range services.Failures {
  781. return compose.ServiceStatus{}, errors.Wrapf(errdefs.ErrNotFound, "can't get service status %s: %s", aws.StringValue(f.Detail), aws.StringValue(f.Reason))
  782. }
  783. service := services.Services[0]
  784. var name string
  785. for _, t := range service.Tags {
  786. if *t.Key == compose.ServiceTag {
  787. name = aws.StringValue(t.Value)
  788. }
  789. }
  790. if name == "" {
  791. return compose.ServiceStatus{}, fmt.Errorf("service %s doesn't have a %s tag", *service.ServiceArn, compose.ServiceTag)
  792. }
  793. targetGroupArns := []string{}
  794. for _, lb := range service.LoadBalancers {
  795. targetGroupArns = append(targetGroupArns, *lb.TargetGroupArn)
  796. }
  797. // getURLwithPortMapping makes 2 queries
  798. // one to get the target groups and another for load balancers
  799. loadBalancers, err := s.getURLWithPortMapping(ctx, targetGroupArns)
  800. if err != nil {
  801. return compose.ServiceStatus{}, err
  802. }
  803. return compose.ServiceStatus{
  804. ID: aws.StringValue(service.ServiceName),
  805. Name: name,
  806. Replicas: int(aws.Int64Value(service.RunningCount)),
  807. Desired: int(aws.Int64Value(service.DesiredCount)),
  808. Publishers: loadBalancers,
  809. }, nil
  810. }
  811. func (s sdk) DescribeServiceTasks(ctx context.Context, cluster string, project string, service string) ([]compose.ContainerSummary, error) {
  812. var summary []compose.ContainerSummary
  813. familly := fmt.Sprintf("%s-%s", project, service)
  814. var token *string
  815. for {
  816. list, err := s.ECS.ListTasks(&ecs.ListTasksInput{
  817. Cluster: aws.String(cluster),
  818. Family: aws.String(familly),
  819. LaunchType: nil,
  820. MaxResults: nil,
  821. NextToken: token,
  822. })
  823. if err != nil {
  824. return nil, err
  825. }
  826. if len(list.TaskArns) == 0 {
  827. break
  828. }
  829. tasks, err := s.ECS.DescribeTasksWithContext(ctx, &ecs.DescribeTasksInput{
  830. Cluster: aws.String(cluster),
  831. Include: aws.StringSlice([]string{"TAGS"}),
  832. Tasks: list.TaskArns,
  833. })
  834. if err != nil {
  835. return nil, err
  836. }
  837. for _, t := range tasks.Tasks {
  838. var project string
  839. var service string
  840. for _, tag := range t.Tags {
  841. switch aws.StringValue(tag.Key) {
  842. case compose.ProjectTag:
  843. project = aws.StringValue(tag.Value)
  844. case compose.ServiceTag:
  845. service = aws.StringValue(tag.Value)
  846. }
  847. }
  848. id, err := arn.Parse(aws.StringValue(t.TaskArn))
  849. if err != nil {
  850. return nil, err
  851. }
  852. summary = append(summary, compose.ContainerSummary{
  853. ID: id.String(),
  854. Name: id.Resource,
  855. Project: project,
  856. Service: service,
  857. State: strings.Title(strings.ToLower(aws.StringValue(t.LastStatus))),
  858. })
  859. }
  860. if list.NextToken == token {
  861. break
  862. }
  863. token = list.NextToken
  864. }
  865. return summary, nil
  866. }
  867. func (s sdk) getURLWithPortMapping(ctx context.Context, targetGroupArns []string) ([]compose.PortPublisher, error) {
  868. if len(targetGroupArns) == 0 {
  869. return nil, nil
  870. }
  871. groups, err := s.ELB.DescribeTargetGroups(&elbv2.DescribeTargetGroupsInput{
  872. TargetGroupArns: aws.StringSlice(targetGroupArns),
  873. })
  874. if err != nil {
  875. return nil, err
  876. }
  877. lbarns := []*string{}
  878. for _, tg := range groups.TargetGroups {
  879. lbarns = append(lbarns, tg.LoadBalancerArns...)
  880. }
  881. lbs, err := s.ELB.DescribeLoadBalancersWithContext(ctx, &elbv2.DescribeLoadBalancersInput{
  882. LoadBalancerArns: lbarns,
  883. })
  884. if err != nil {
  885. return nil, err
  886. }
  887. filterLB := func(arn *string, lbs []*elbv2.LoadBalancer) *elbv2.LoadBalancer {
  888. if aws.StringValue(arn) == "" {
  889. // load balancer arn is nil/""
  890. return nil
  891. }
  892. for _, lb := range lbs {
  893. if aws.StringValue(lb.LoadBalancerArn) == aws.StringValue(arn) {
  894. return lb
  895. }
  896. }
  897. return nil
  898. }
  899. loadBalancers := []compose.PortPublisher{}
  900. for _, tg := range groups.TargetGroups {
  901. for _, lbarn := range tg.LoadBalancerArns {
  902. lb := filterLB(lbarn, lbs.LoadBalancers)
  903. if lb == nil {
  904. continue
  905. }
  906. loadBalancers = append(loadBalancers, compose.PortPublisher{
  907. URL: fmt.Sprintf("%s:%d", aws.StringValue(lb.DNSName), aws.Int64Value(tg.Port)),
  908. TargetPort: int(aws.Int64Value(tg.Port)),
  909. PublishedPort: int(aws.Int64Value(tg.Port)),
  910. Protocol: strings.ToLower(aws.StringValue(tg.Protocol)),
  911. })
  912. }
  913. }
  914. return loadBalancers, nil
  915. }
  916. func (s sdk) ListTasks(ctx context.Context, cluster string, family string) ([]string, error) {
  917. var token *string
  918. var arns []string
  919. for {
  920. response, err := s.ECS.ListTasksWithContext(ctx, &ecs.ListTasksInput{
  921. Cluster: aws.String(cluster),
  922. Family: aws.String(family),
  923. })
  924. if err != nil {
  925. return nil, err
  926. }
  927. for _, arn := range response.TaskArns {
  928. arns = append(arns, *arn)
  929. }
  930. if token == response.NextToken {
  931. return arns, nil
  932. }
  933. token = response.NextToken
  934. }
  935. }
  936. func (s sdk) GetPublicIPs(ctx context.Context, interfaces ...string) (map[string]string, error) {
  937. var token *string
  938. publicIPs := map[string]string{}
  939. for {
  940. response, err := s.EC2.DescribeNetworkInterfaces(&ec2.DescribeNetworkInterfacesInput{
  941. NetworkInterfaceIds: aws.StringSlice(interfaces),
  942. })
  943. if err != nil {
  944. return nil, err
  945. }
  946. for _, interf := range response.NetworkInterfaces {
  947. if interf.Association != nil {
  948. publicIPs[aws.StringValue(interf.NetworkInterfaceId)] = aws.StringValue(interf.Association.PublicIp)
  949. }
  950. }
  951. if token == response.NextToken {
  952. return publicIPs, nil
  953. }
  954. token = response.NextToken
  955. }
  956. }
  957. func (s sdk) ResolveLoadBalancer(ctx context.Context, nameOrArn string) (awsResource, string, string, []awsResource, error) {
  958. logrus.Debug("Check if LoadBalancer exists: ", nameOrArn)
  959. var arns []*string
  960. var names []*string
  961. if arn.IsARN(nameOrArn) {
  962. arns = append(arns, aws.String(nameOrArn))
  963. } else {
  964. names = append(names, aws.String(nameOrArn))
  965. }
  966. lbs, err := s.ELB.DescribeLoadBalancersWithContext(ctx, &elbv2.DescribeLoadBalancersInput{
  967. LoadBalancerArns: arns,
  968. Names: names,
  969. })
  970. if err != nil {
  971. return nil, "", "", nil, err
  972. }
  973. if len(lbs.LoadBalancers) == 0 {
  974. return nil, "", "", nil, errors.Wrapf(errdefs.ErrNotFound, "load balancer %q does not exist", nameOrArn)
  975. }
  976. it := lbs.LoadBalancers[0]
  977. var subNets []awsResource
  978. for _, az := range it.AvailabilityZones {
  979. subNets = append(subNets, existingAWSResource{
  980. id: aws.StringValue(az.SubnetId),
  981. })
  982. }
  983. return existingAWSResource{
  984. arn: aws.StringValue(it.LoadBalancerArn),
  985. id: aws.StringValue(it.LoadBalancerName),
  986. }, aws.StringValue(it.Type), aws.StringValue(it.VpcId), subNets, nil
  987. }
  988. func (s sdk) GetLoadBalancerURL(ctx context.Context, arn string) (string, error) {
  989. logrus.Debug("Retrieve load balancer URL: ", arn)
  990. lbs, err := s.ELB.DescribeLoadBalancersWithContext(ctx, &elbv2.DescribeLoadBalancersInput{
  991. LoadBalancerArns: []*string{aws.String(arn)},
  992. })
  993. if err != nil {
  994. return "", err
  995. }
  996. dnsName := aws.StringValue(lbs.LoadBalancers[0].DNSName)
  997. if dnsName == "" {
  998. return "", fmt.Errorf("Load balancer %s doesn't have a dns name", aws.StringValue(lbs.LoadBalancers[0].LoadBalancerArn))
  999. }
  1000. return dnsName, nil
  1001. }
  1002. func (s sdk) GetParameter(ctx context.Context, name string) (string, error) {
  1003. parameter, err := s.SSM.GetParameterWithContext(ctx, &ssm.GetParameterInput{
  1004. Name: aws.String(name),
  1005. })
  1006. if err != nil {
  1007. return "", err
  1008. }
  1009. value := *parameter.Parameter.Value
  1010. var ami struct {
  1011. SchemaVersion int `json:"schema_version"`
  1012. ImageName string `json:"image_name"`
  1013. ImageID string `json:"image_id"`
  1014. OS string `json:"os"`
  1015. ECSRuntimeVersion string `json:"ecs_runtime_verion"`
  1016. ECSAgentVersion string `json:"ecs_agent_version"`
  1017. }
  1018. err = json.Unmarshal([]byte(value), &ami)
  1019. if err != nil {
  1020. return "", err
  1021. }
  1022. return ami.ImageID, nil
  1023. }
  1024. func (s sdk) SecurityGroupExists(ctx context.Context, sg string) (bool, error) {
  1025. desc, err := s.EC2.DescribeSecurityGroupsWithContext(ctx, &ec2.DescribeSecurityGroupsInput{
  1026. GroupIds: aws.StringSlice([]string{sg}),
  1027. })
  1028. if err != nil {
  1029. return false, err
  1030. }
  1031. return len(desc.SecurityGroups) > 0, nil
  1032. }
  1033. func (s sdk) DeleteCapacityProvider(ctx context.Context, arn string) error {
  1034. _, err := s.ECS.DeleteCapacityProvider(&ecs.DeleteCapacityProviderInput{
  1035. CapacityProvider: aws.String(arn),
  1036. })
  1037. return err
  1038. }
  1039. func (s sdk) DeleteAutoscalingGroup(ctx context.Context, arn string) error {
  1040. _, err := s.AG.DeleteAutoScalingGroup(&autoscaling.DeleteAutoScalingGroupInput{
  1041. AutoScalingGroupName: aws.String(arn),
  1042. ForceDelete: aws.Bool(true),
  1043. })
  1044. return err
  1045. }
  1046. func (s sdk) ResolveFileSystem(ctx context.Context, id string) (awsResource, error) {
  1047. desc, err := s.EFS.DescribeFileSystemsWithContext(ctx, &efs.DescribeFileSystemsInput{
  1048. FileSystemId: aws.String(id),
  1049. })
  1050. if err != nil {
  1051. return nil, err
  1052. }
  1053. if len(desc.FileSystems) == 0 {
  1054. return nil, errors.Wrapf(errdefs.ErrNotFound, "EFS file system %q doesn't exist", id)
  1055. }
  1056. it := desc.FileSystems[0]
  1057. return existingAWSResource{
  1058. arn: aws.StringValue(it.FileSystemArn),
  1059. id: aws.StringValue(it.FileSystemId),
  1060. }, nil
  1061. }
  1062. func (s sdk) ListFileSystems(ctx context.Context, tags map[string]string) ([]awsResource, error) {
  1063. var results []awsResource
  1064. var token *string
  1065. for {
  1066. desc, err := s.EFS.DescribeFileSystemsWithContext(ctx, &efs.DescribeFileSystemsInput{
  1067. Marker: token,
  1068. })
  1069. if err != nil {
  1070. return nil, err
  1071. }
  1072. for _, filesystem := range desc.FileSystems {
  1073. if containsAll(filesystem.Tags, tags) {
  1074. results = append(results, existingAWSResource{
  1075. arn: aws.StringValue(filesystem.FileSystemArn),
  1076. id: aws.StringValue(filesystem.FileSystemId),
  1077. })
  1078. }
  1079. }
  1080. if desc.NextMarker == token {
  1081. return results, nil
  1082. }
  1083. token = desc.NextMarker
  1084. }
  1085. }
  1086. func containsAll(tags []*efs.Tag, required map[string]string) bool {
  1087. TAGS:
  1088. for key, value := range required {
  1089. for _, t := range tags {
  1090. if aws.StringValue(t.Key) == key && aws.StringValue(t.Value) == value {
  1091. continue TAGS
  1092. }
  1093. }
  1094. return false
  1095. }
  1096. return true
  1097. }
  1098. func (s sdk) CreateFileSystem(ctx context.Context, tags map[string]string, options VolumeCreateOptions) (awsResource, error) {
  1099. var efsTags []*efs.Tag
  1100. for k, v := range tags {
  1101. efsTags = append(efsTags, &efs.Tag{
  1102. Key: aws.String(k),
  1103. Value: aws.String(v),
  1104. })
  1105. }
  1106. var (
  1107. k *string
  1108. p *string
  1109. f *float64
  1110. t *string
  1111. )
  1112. if options.ProvisionedThroughputInMibps > 1 {
  1113. f = aws.Float64(options.ProvisionedThroughputInMibps)
  1114. }
  1115. if options.KmsKeyID != "" {
  1116. k = aws.String(options.KmsKeyID)
  1117. }
  1118. if options.PerformanceMode != "" {
  1119. p = aws.String(options.PerformanceMode)
  1120. }
  1121. if options.ThroughputMode != "" {
  1122. t = aws.String(options.ThroughputMode)
  1123. }
  1124. res, err := s.EFS.CreateFileSystemWithContext(ctx, &efs.CreateFileSystemInput{
  1125. Encrypted: aws.Bool(true),
  1126. KmsKeyId: k,
  1127. PerformanceMode: p,
  1128. ProvisionedThroughputInMibps: f,
  1129. ThroughputMode: t,
  1130. Tags: efsTags,
  1131. })
  1132. if err != nil {
  1133. return nil, err
  1134. }
  1135. return existingAWSResource{
  1136. id: aws.StringValue(res.FileSystemId),
  1137. arn: aws.StringValue(res.FileSystemArn),
  1138. }, nil
  1139. }
  1140. func (s sdk) DeleteFileSystem(ctx context.Context, id string) error {
  1141. _, err := s.EFS.DeleteFileSystemWithContext(ctx, &efs.DeleteFileSystemInput{
  1142. FileSystemId: aws.String(id),
  1143. })
  1144. return err
  1145. }