2019-02-11 13:36:04 -05:00
|
|
|
package bsu
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"fmt"
|
|
|
|
"log"
|
|
|
|
|
2020-08-25 19:02:11 -04:00
|
|
|
"github.com/antihax/optional"
|
2020-12-17 16:29:25 -05:00
|
|
|
"github.com/hashicorp/packer-plugin-sdk/multistep"
|
|
|
|
packersdk "github.com/hashicorp/packer-plugin-sdk/packer"
|
2019-02-11 13:36:04 -05:00
|
|
|
osccommon "github.com/hashicorp/packer/builder/osc/common"
|
2020-08-20 21:37:09 -04:00
|
|
|
"github.com/outscale/osc-sdk-go/osc"
|
2019-02-11 13:36:04 -05:00
|
|
|
)
|
|
|
|
|
|
|
|
type stepCreateOMI struct {
|
2020-08-25 19:02:11 -04:00
|
|
|
image *osc.Image
|
|
|
|
RawRegion string
|
2019-02-11 13:36:04 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
func (s *stepCreateOMI) Run(ctx context.Context, state multistep.StateBag) multistep.StepAction {
|
|
|
|
config := state.Get("config").(*Config)
|
2020-08-25 19:02:11 -04:00
|
|
|
oscconn := state.Get("osc").(*osc.APIClient)
|
2020-08-20 21:37:09 -04:00
|
|
|
vm := state.Get("vm").(osc.Vm)
|
2020-11-19 14:54:31 -05:00
|
|
|
ui := state.Get("ui").(packersdk.Ui)
|
2019-02-11 13:36:04 -05:00
|
|
|
|
|
|
|
// Create the image
|
|
|
|
omiName := config.OMIName
|
|
|
|
|
2019-02-12 16:15:38 -05:00
|
|
|
ui.Say(fmt.Sprintf("Creating OMI %s from vm %s", omiName, vm.VmId))
|
2020-08-25 19:02:11 -04:00
|
|
|
createOpts := osc.CreateImageRequest{
|
2019-02-11 13:36:04 -05:00
|
|
|
VmId: vm.VmId,
|
|
|
|
ImageName: omiName,
|
2020-08-25 19:02:11 -04:00
|
|
|
BlockDeviceMappings: config.BlockDevices.BuildOscOMIDevices(),
|
2019-02-11 13:36:04 -05:00
|
|
|
}
|
|
|
|
|
2020-08-25 19:02:11 -04:00
|
|
|
resp, _, err := oscconn.ImageApi.CreateImage(context.Background(), &osc.CreateImageOpts{
|
|
|
|
CreateImageRequest: optional.NewInterface(createOpts),
|
|
|
|
})
|
|
|
|
if err != nil || resp.Image.ImageId == "" {
|
2019-02-11 13:36:04 -05:00
|
|
|
err := fmt.Errorf("Error creating OMI: %s", err)
|
|
|
|
state.Put("error", err)
|
|
|
|
ui.Error(err.Error())
|
|
|
|
return multistep.ActionHalt
|
|
|
|
}
|
|
|
|
|
2020-08-25 19:02:11 -04:00
|
|
|
image := resp.Image
|
2019-02-11 13:36:04 -05:00
|
|
|
|
|
|
|
// Set the OMI ID in the state
|
|
|
|
ui.Message(fmt.Sprintf("OMI: %s", image.ImageId))
|
|
|
|
omis := make(map[string]string)
|
2020-08-25 19:02:11 -04:00
|
|
|
omis[s.RawRegion] = image.ImageId
|
2019-02-11 13:36:04 -05:00
|
|
|
state.Put("omis", omis)
|
|
|
|
|
|
|
|
// Wait for the image to become ready
|
|
|
|
ui.Say("Waiting for OMI to become ready...")
|
2020-08-25 19:02:11 -04:00
|
|
|
if err := osccommon.WaitUntilOscImageAvailable(oscconn, image.ImageId); err != nil {
|
2019-02-11 13:36:04 -05:00
|
|
|
log.Printf("Error waiting for OMI: %s", err)
|
2020-08-25 19:02:11 -04:00
|
|
|
imagesResp, _, err := oscconn.ImageApi.ReadImages(context.Background(), &osc.ReadImagesOpts{
|
|
|
|
ReadImagesRequest: optional.NewInterface(osc.ReadImagesRequest{
|
|
|
|
Filters: osc.FiltersImage{
|
|
|
|
ImageIds: []string{image.ImageId},
|
|
|
|
},
|
|
|
|
}),
|
2019-02-11 13:36:04 -05:00
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
log.Printf("Unable to determine reason waiting for OMI failed: %s", err)
|
2020-08-25 19:02:11 -04:00
|
|
|
err = fmt.Errorf("Unknown error waiting for OMI")
|
2019-02-11 13:36:04 -05:00
|
|
|
} else {
|
2020-08-25 19:02:11 -04:00
|
|
|
stateReason := imagesResp.Images[0].StateComment
|
2019-02-11 13:36:04 -05:00
|
|
|
err = fmt.Errorf("Error waiting for OMI. Reason: %s", stateReason)
|
|
|
|
}
|
|
|
|
|
|
|
|
state.Put("error", err)
|
|
|
|
ui.Error(err.Error())
|
|
|
|
return multistep.ActionHalt
|
|
|
|
}
|
|
|
|
|
2020-08-25 19:02:11 -04:00
|
|
|
imagesResp, _, err := oscconn.ImageApi.ReadImages(context.Background(), &osc.ReadImagesOpts{
|
|
|
|
ReadImagesRequest: optional.NewInterface(osc.ReadImagesRequest{
|
|
|
|
Filters: osc.FiltersImage{
|
|
|
|
ImageIds: []string{image.ImageId},
|
|
|
|
},
|
|
|
|
}),
|
2019-02-11 13:36:04 -05:00
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
err := fmt.Errorf("Error searching for OMI: %s", err)
|
|
|
|
state.Put("error", err)
|
|
|
|
ui.Error(err.Error())
|
|
|
|
return multistep.ActionHalt
|
|
|
|
}
|
2020-08-25 19:02:11 -04:00
|
|
|
s.image = &imagesResp.Images[0]
|
2019-02-11 13:36:04 -05:00
|
|
|
|
|
|
|
snapshots := make(map[string][]string)
|
2020-08-25 19:02:11 -04:00
|
|
|
for _, blockDeviceMapping := range imagesResp.Images[0].BlockDeviceMappings {
|
2019-02-11 13:36:04 -05:00
|
|
|
if blockDeviceMapping.Bsu.SnapshotId != "" {
|
2020-08-25 19:02:11 -04:00
|
|
|
snapshots[s.RawRegion] = append(snapshots[s.RawRegion], blockDeviceMapping.Bsu.SnapshotId)
|
2019-02-11 13:36:04 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
state.Put("snapshots", snapshots)
|
|
|
|
|
|
|
|
return multistep.ActionContinue
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *stepCreateOMI) Cleanup(state multistep.StateBag) {
|
|
|
|
if s.image == nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
_, cancelled := state.GetOk(multistep.StateCancelled)
|
|
|
|
_, halted := state.GetOk(multistep.StateHalted)
|
|
|
|
if !cancelled && !halted {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2020-08-25 19:02:11 -04:00
|
|
|
oscconn := state.Get("osc").(*osc.APIClient)
|
2020-11-19 14:54:31 -05:00
|
|
|
ui := state.Get("ui").(packersdk.Ui)
|
2019-02-11 13:36:04 -05:00
|
|
|
|
|
|
|
ui.Say("Deregistering the OMI because cancellation or error...")
|
2020-08-25 19:02:11 -04:00
|
|
|
DeleteOpts := osc.DeleteImageRequest{ImageId: s.image.ImageId}
|
|
|
|
if _, _, err := oscconn.ImageApi.DeleteImage(context.Background(), &osc.DeleteImageOpts{
|
|
|
|
DeleteImageRequest: optional.NewInterface(DeleteOpts),
|
|
|
|
}); err != nil {
|
2019-02-11 13:36:04 -05:00
|
|
|
ui.Error(fmt.Sprintf("Error Deleting OMI, may still be around: %s", err))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|