packer-cn/builder/ucloud/uhost/step_copy_image.go

144 lines
4.3 KiB
Go
Raw Normal View History

2019-06-13 03:16:49 -04:00
package uhost
import (
"context"
"fmt"
2019-10-12 04:46:21 -04:00
ucloudcommon "github.com/hashicorp/packer/builder/ucloud/common"
2019-06-13 03:16:49 -04:00
"github.com/hashicorp/packer/common/retry"
2019-06-19 09:32:33 -04:00
"strings"
2019-06-13 03:16:49 -04:00
"time"
"github.com/hashicorp/packer/helper/multistep"
"github.com/hashicorp/packer/packer"
"github.com/ucloud/ucloud-sdk-go/ucloud"
)
type stepCopyUCloudImage struct {
2019-10-18 02:52:20 -04:00
ImageDestinations []ucloudcommon.ImageDestination
RegionId string
ProjectId string
WaitImageReadyTimeout int
2019-06-13 03:16:49 -04:00
}
func (s *stepCopyUCloudImage) Run(ctx context.Context, state multistep.StateBag) multistep.StepAction {
if len(s.ImageDestinations) == 0 {
return multistep.ActionContinue
}
2019-10-12 04:46:21 -04:00
client := state.Get("client").(*ucloudcommon.UCloudClient)
conn := client.UHostConn
2019-06-13 03:16:49 -04:00
ui := state.Get("ui").(packer.Ui)
srcImageId := state.Get("image_id").(string)
2019-10-12 04:46:21 -04:00
artifactImages := state.Get("ucloud_images").(*ucloudcommon.ImageInfoSet)
expectedImages := ucloudcommon.NewImageInfoSet(nil)
2019-07-26 05:27:58 -04:00
ui.Say(fmt.Sprintf("Copying images from %q...", srcImageId))
2019-06-19 09:32:33 -04:00
for _, v := range s.ImageDestinations {
if v.ProjectId == s.ProjectId && v.Region == s.RegionId {
2019-06-13 03:16:49 -04:00
continue
}
req := conn.NewCopyCustomImageRequest()
2019-06-19 09:32:33 -04:00
req.TargetProjectId = ucloud.String(v.ProjectId)
req.TargetRegion = ucloud.String(v.Region)
2019-06-13 03:16:49 -04:00
req.SourceImageId = ucloud.String(srcImageId)
2019-06-19 09:32:33 -04:00
req.TargetImageName = ucloud.String(v.Name)
req.TargetImageDescription = ucloud.String(v.Description)
2019-06-13 03:16:49 -04:00
resp, err := conn.CopyCustomImage(req)
if err != nil {
2019-10-12 04:46:21 -04:00
return ucloudcommon.Halt(state, err, fmt.Sprintf("Error on copying image %q to %s:%s", srcImageId, v.ProjectId, v.Region))
2019-06-13 03:16:49 -04:00
}
2019-10-12 04:46:21 -04:00
image := ucloudcommon.ImageInfo{
2019-06-19 09:32:33 -04:00
Region: v.Region,
ProjectId: v.ProjectId,
2019-06-13 03:16:49 -04:00
ImageId: resp.TargetImageId,
}
expectedImages.Set(image)
artifactImages.Set(image)
ui.Message(fmt.Sprintf("Copying image from %s:%s:%s to %s:%s:%s)",
2019-06-19 09:32:33 -04:00
s.ProjectId, s.RegionId, srcImageId, v.ProjectId, v.Region, resp.TargetImageId))
2019-06-13 03:16:49 -04:00
}
2019-07-26 05:03:57 -04:00
ui.Message("Waiting for the copied images to become available...")
2019-06-13 03:16:49 -04:00
err := retry.Config{
2019-10-18 02:52:20 -04:00
StartTimeout: time.Duration(s.WaitImageReadyTimeout) * time.Second,
ShouldRetry: func(err error) bool {
return ucloudcommon.IsNotCompleteError(err)
},
RetryDelay: (&retry.Backoff{InitialBackoff: 2 * time.Second, MaxBackoff: 12 * time.Second, Multiplier: 2}).Linear,
2019-06-13 03:16:49 -04:00
}.Run(ctx, func(ctx context.Context) error {
for _, v := range expectedImages.GetAll() {
2019-10-12 04:46:21 -04:00
imageSet, err := client.DescribeImageByInfo(v.ProjectId, v.Region, v.ImageId)
2019-06-13 03:16:49 -04:00
if err != nil {
2019-06-19 09:32:33 -04:00
return fmt.Errorf("reading %s:%s:%s failed, %s", v.ProjectId, v.Region, v.ImageId, err)
2019-06-13 03:16:49 -04:00
}
2019-10-12 04:46:21 -04:00
if imageSet.State == ucloudcommon.ImageStateAvailable {
2019-06-13 03:16:49 -04:00
expectedImages.Remove(v.Id())
continue
}
}
if len(expectedImages.GetAll()) != 0 {
2019-10-12 04:46:21 -04:00
return ucloudcommon.NewNotCompleteError("copying image")
2019-06-13 03:16:49 -04:00
}
return nil
})
if err != nil {
2019-06-19 09:32:33 -04:00
var s []string
for _, v := range expectedImages.GetAll() {
s = append(s, fmt.Sprintf("%s:%s:%s", v.ProjectId, v.Region, v.ImageId))
}
2019-10-12 04:46:21 -04:00
return ucloudcommon.Halt(state, err, fmt.Sprintf("Error on waiting for copying images %q to become available", strings.Join(s, ",")))
2019-06-13 03:16:49 -04:00
}
2019-06-19 09:32:33 -04:00
ui.Message(fmt.Sprintf("Copying image complete"))
2019-06-13 03:16:49 -04:00
return multistep.ActionContinue
}
func (s *stepCopyUCloudImage) Cleanup(state multistep.StateBag) {
_, cancelled := state.GetOk(multistep.StateCancelled)
_, halted := state.GetOk(multistep.StateHalted)
if !cancelled && !halted {
return
}
srcImageId := state.Get("image_id").(string)
2019-10-12 04:46:21 -04:00
ucloudImages := state.Get("ucloud_images").(*ucloudcommon.ImageInfoSet)
2019-06-13 03:16:49 -04:00
imageInfos := ucloudImages.GetAll()
if len(imageInfos) == 0 {
return
} else if len(imageInfos) == 1 && imageInfos[0].ImageId == srcImageId {
return
}
ui := state.Get("ui").(packer.Ui)
2019-10-12 04:46:21 -04:00
client := state.Get("client").(*ucloudcommon.UCloudClient)
conn := client.UHostConn
ui.Say(fmt.Sprintf("Deleting copied image because of cancellation or error..."))
2019-06-13 03:16:49 -04:00
for _, v := range imageInfos {
if v.ImageId == srcImageId {
continue
}
req := conn.NewTerminateCustomImageRequest()
req.ProjectId = ucloud.String(v.ProjectId)
req.Region = ucloud.String(v.Region)
req.ImageId = ucloud.String(v.ImageId)
_, err := conn.TerminateCustomImage(req)
if err != nil {
ui.Error(fmt.Sprintf("Error on deleting copied image %q", v.ImageId))
}
}
2019-06-19 09:32:33 -04:00
ui.Message("Deleting copied image complete")
2019-06-13 03:16:49 -04:00
}