2020-08-18 11:38:23 +02:00
|
|
|
/*
|
2020-09-22 12:13:00 +02:00
|
|
|
Copyright 2020 Docker Compose CLI authors
|
2020-08-18 11:38:23 +02:00
|
|
|
|
|
|
|
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.
|
|
|
|
*/
|
|
|
|
|
2020-08-17 17:48:52 +02:00
|
|
|
package ecs
|
2020-04-29 15:26:11 +02:00
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"fmt"
|
|
|
|
"sort"
|
2020-04-30 16:14:36 +02:00
|
|
|
"strings"
|
2020-04-30 15:42:34 +02:00
|
|
|
"time"
|
2020-04-29 15:26:11 +02:00
|
|
|
|
2020-08-21 17:24:53 +02:00
|
|
|
"github.com/docker/compose-cli/progress"
|
2020-08-18 11:38:23 +02:00
|
|
|
|
2020-04-29 15:26:11 +02:00
|
|
|
"github.com/aws/aws-sdk-go/aws"
|
|
|
|
)
|
|
|
|
|
2020-10-02 14:08:23 +02:00
|
|
|
func (b *ecsAPIService) WaitStackCompletion(ctx context.Context, name string, operation int, ignored ...string) error { //nolint:gocyclo
|
2020-04-30 15:42:34 +02:00
|
|
|
knownEvents := map[string]struct{}{}
|
2020-10-02 14:08:23 +02:00
|
|
|
for _, id := range ignored {
|
|
|
|
knownEvents[id] = struct{}{}
|
|
|
|
}
|
|
|
|
|
2020-08-14 12:01:22 +02:00
|
|
|
// progress writer
|
|
|
|
w := progress.ContextWriter(ctx)
|
2020-04-30 15:42:34 +02:00
|
|
|
// Get the unique Stack ID so we can collect events without getting some from previous deployments with same name
|
2020-10-12 16:21:48 +02:00
|
|
|
stackID, err := b.aws.GetStackID(ctx, name)
|
2020-04-30 15:42:34 +02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
ticker := time.NewTicker(1 * time.Second)
|
2020-04-30 16:14:36 +02:00
|
|
|
done := make(chan bool)
|
2020-04-30 15:42:34 +02:00
|
|
|
go func() {
|
2020-10-12 16:21:48 +02:00
|
|
|
b.aws.WaitStackComplete(ctx, stackID, operation) //nolint:errcheck
|
2020-04-30 15:42:34 +02:00
|
|
|
ticker.Stop()
|
2020-04-30 16:14:36 +02:00
|
|
|
done <- true
|
2020-04-30 15:42:34 +02:00
|
|
|
}()
|
|
|
|
|
|
|
|
var completed bool
|
2020-04-30 16:14:36 +02:00
|
|
|
var stackErr error
|
2020-04-30 15:42:34 +02:00
|
|
|
for !completed {
|
|
|
|
select {
|
2020-04-30 16:14:36 +02:00
|
|
|
case <-done:
|
2020-04-30 15:42:34 +02:00
|
|
|
completed = true
|
|
|
|
case <-ticker.C:
|
|
|
|
}
|
2020-10-12 16:21:48 +02:00
|
|
|
events, err := b.aws.DescribeStackEvents(ctx, stackID)
|
2020-04-29 15:26:11 +02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
sort.Slice(events, func(i, j int) bool {
|
|
|
|
return events[i].Timestamp.Before(*events[j].Timestamp)
|
|
|
|
})
|
|
|
|
|
|
|
|
for _, event := range events {
|
2020-04-30 15:42:34 +02:00
|
|
|
if _, ok := knownEvents[*event.EventId]; ok {
|
2020-04-29 15:26:11 +02:00
|
|
|
continue
|
|
|
|
}
|
2020-04-30 15:42:34 +02:00
|
|
|
knownEvents[*event.EventId] = struct{}{}
|
2020-04-29 15:26:11 +02:00
|
|
|
|
2020-06-25 08:14:54 +02:00
|
|
|
resource := aws.StringValue(event.LogicalResourceId)
|
2020-10-02 11:05:06 +02:00
|
|
|
reason := shortenMessage(
|
|
|
|
aws.StringValue(event.ResourceStatusReason))
|
2020-04-30 16:14:36 +02:00
|
|
|
status := aws.StringValue(event.ResourceStatus)
|
2020-08-11 10:43:11 +02:00
|
|
|
progressStatus := progress.Working
|
2020-09-30 14:52:53 +02:00
|
|
|
|
2020-10-02 11:05:06 +02:00
|
|
|
switch status {
|
2020-08-11 10:43:11 +02:00
|
|
|
case "CREATE_COMPLETE":
|
2020-08-18 16:56:42 +02:00
|
|
|
if operation == stackCreate {
|
2020-08-11 10:43:11 +02:00
|
|
|
progressStatus = progress.Done
|
|
|
|
}
|
|
|
|
case "UPDATE_COMPLETE":
|
2020-08-18 16:56:42 +02:00
|
|
|
if operation == stackUpdate {
|
2020-08-11 10:43:11 +02:00
|
|
|
progressStatus = progress.Done
|
|
|
|
}
|
|
|
|
case "DELETE_COMPLETE":
|
2020-08-18 16:56:42 +02:00
|
|
|
if operation == stackDelete {
|
2020-08-11 10:43:11 +02:00
|
|
|
progressStatus = progress.Done
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
if strings.HasSuffix(status, "_FAILED") {
|
|
|
|
progressStatus = progress.Error
|
|
|
|
if stackErr == nil {
|
2020-08-18 16:56:42 +02:00
|
|
|
operation = stackDelete
|
2020-08-11 10:43:11 +02:00
|
|
|
stackErr = fmt.Errorf(reason)
|
|
|
|
}
|
|
|
|
}
|
2020-04-30 16:14:36 +02:00
|
|
|
}
|
2020-08-14 12:01:22 +02:00
|
|
|
w.Event(progress.Event{
|
2020-08-11 10:43:11 +02:00
|
|
|
ID: resource,
|
|
|
|
Status: progressStatus,
|
2020-10-06 14:05:08 +02:00
|
|
|
StatusText: fmt.Sprintf("%s %s", status, reason),
|
2020-08-11 10:43:11 +02:00
|
|
|
})
|
2020-04-29 15:26:11 +02:00
|
|
|
}
|
2020-09-30 14:52:53 +02:00
|
|
|
if operation != stackCreate || stackErr != nil {
|
|
|
|
continue
|
|
|
|
}
|
2020-10-02 11:05:06 +02:00
|
|
|
if err := b.checkStackState(ctx, name); err != nil {
|
2020-10-12 16:21:48 +02:00
|
|
|
if e := b.aws.DeleteStack(ctx, name); e != nil {
|
2020-10-01 10:31:37 +02:00
|
|
|
return e
|
|
|
|
}
|
2020-09-30 14:52:53 +02:00
|
|
|
stackErr = err
|
|
|
|
operation = stackDelete
|
2020-10-02 11:05:06 +02:00
|
|
|
reason := shortenMessage(err.Error())
|
2020-09-30 14:52:53 +02:00
|
|
|
w.Event(progress.Event{
|
|
|
|
ID: name,
|
|
|
|
Status: progress.Error,
|
|
|
|
StatusText: reason,
|
|
|
|
})
|
|
|
|
}
|
2020-04-29 15:26:11 +02:00
|
|
|
}
|
2020-08-11 10:43:11 +02:00
|
|
|
|
2020-04-30 16:14:36 +02:00
|
|
|
return stackErr
|
2020-04-29 15:26:11 +02:00
|
|
|
}
|
2020-10-02 11:05:06 +02:00
|
|
|
|
|
|
|
func shortenMessage(message string) string {
|
|
|
|
if len(message) < 30 {
|
|
|
|
return message
|
|
|
|
}
|
|
|
|
return message[:30] + "..."
|
|
|
|
}
|