2019-06-13 03:16:49 -04:00
|
|
|
package uhost
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"fmt"
|
|
|
|
"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 {
|
|
|
|
ImageDestinations []ImageDestination
|
|
|
|
RegionId string
|
|
|
|
ProjectId string
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *stepCopyUCloudImage) Run(ctx context.Context, state multistep.StateBag) multistep.StepAction {
|
|
|
|
if len(s.ImageDestinations) == 0 {
|
|
|
|
return multistep.ActionContinue
|
|
|
|
}
|
|
|
|
|
|
|
|
client := state.Get("client").(*UCloudClient)
|
|
|
|
conn := client.uhostconn
|
|
|
|
ui := state.Get("ui").(packer.Ui)
|
|
|
|
|
|
|
|
srcImageId := state.Get("image_id").(string)
|
|
|
|
artifactImages := state.Get("ucloud_images").(*imageInfoSet)
|
|
|
|
expectedImages := newImageInfoSet(nil)
|
2019-07-26 05:03:57 -04:00
|
|
|
ui.Say(fmt.Sprintf("Copying images form %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-06-19 09:32:33 -04:00
|
|
|
return 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
|
|
|
}
|
|
|
|
|
|
|
|
image := 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{
|
|
|
|
Tries: 200,
|
|
|
|
ShouldRetry: func(err error) bool { return isNotCompleteError(err) },
|
|
|
|
RetryDelay: (&retry.Backoff{InitialBackoff: 2 * time.Second, MaxBackoff: 12 * time.Second, Multiplier: 2}).Linear,
|
|
|
|
}.Run(ctx, func(ctx context.Context) error {
|
|
|
|
for _, v := range expectedImages.GetAll() {
|
|
|
|
imageSet, err := client.describeImageByInfo(v.ProjectId, v.Region, v.ImageId)
|
|
|
|
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-06-13 08:17:08 -04:00
|
|
|
if imageSet.State == imageStateAvailable {
|
2019-06-13 03:16:49 -04:00
|
|
|
expectedImages.Remove(v.Id())
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(expectedImages.GetAll()) != 0 {
|
|
|
|
return newNotCompleteError("copying image")
|
|
|
|
}
|
|
|
|
|
|
|
|
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-06-28 00:03:11 -04:00
|
|
|
return 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)
|
|
|
|
ucloudImages := state.Get("ucloud_images").(*imageInfoSet)
|
|
|
|
imageInfos := ucloudImages.GetAll()
|
|
|
|
if len(imageInfos) == 0 {
|
|
|
|
return
|
|
|
|
} else if len(imageInfos) == 1 && imageInfos[0].ImageId == srcImageId {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
ui := state.Get("ui").(packer.Ui)
|
|
|
|
client := state.Get("client").(*UCloudClient)
|
|
|
|
conn := client.uhostconn
|
2019-06-28 00:04:36 -04:00
|
|
|
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
|
|
|
}
|