package yandex import ( "context" "errors" "fmt" "log" "github.com/hashicorp/packer/packer-plugin-sdk/multistep" packersdk "github.com/hashicorp/packer/packer-plugin-sdk/packer" "github.com/hashicorp/packer/packer-plugin-sdk/packerbuilderdata" "github.com/yandex-cloud/go-genproto/yandex/cloud/compute/v1" ycsdk "github.com/yandex-cloud/go-sdk" ) type stepCreateImage struct { GeneratedData *packerbuilderdata.GeneratedData } func (s *stepCreateImage) Run(ctx context.Context, state multistep.StateBag) multistep.StepAction { sdk := state.Get("sdk").(*ycsdk.SDK) ui := state.Get("ui").(packersdk.Ui) c := state.Get("config").(*Config) diskID := state.Get("disk_id").(string) ui.Say(fmt.Sprintf("Creating image: %v", c.ImageName)) ctx, cancel := context.WithTimeout(ctx, c.StateTimeout) defer cancel() op, err := sdk.WrapOperation(sdk.Compute().Image().Create(ctx, &compute.CreateImageRequest{ FolderId: c.TargetImageFolderID, Name: c.ImageName, Family: c.ImageFamily, Description: c.ImageDescription, Labels: c.ImageLabels, MinDiskSize: toBytes(c.ImageMinDiskSizeGb), ProductIds: c.ImageProductIDs, Source: &compute.CreateImageRequest_DiskId{ DiskId: diskID, }, })) if err != nil { return StepHaltWithError(state, fmt.Errorf("Error creating image: %s", err)) } ui.Say("Waiting for image to complete...") if err := op.Wait(ctx); err != nil { return StepHaltWithError(state, fmt.Errorf("Error waiting for image: %s", err)) } resp, err := op.Response() if err != nil { return StepHaltWithError(state, err) } image, ok := resp.(*compute.Image) if !ok { return StepHaltWithError(state, errors.New("API call response doesn't contain Compute Image")) } log.Printf("Image ID: %s", image.Id) log.Printf("Image Name: %s", image.Name) log.Printf("Image Family: %s", image.Family) log.Printf("Image Description: %s", image.Description) log.Printf("Image Storage size: %d", image.StorageSize) state.Put("image", image) // provision generated_data from declared in Builder.Prepare func // see doc https://www.packer.io/docs/extending/custom-builders#build-variables for details s.GeneratedData.Put("ImageID", image.Id) s.GeneratedData.Put("ImageName", image.Name) s.GeneratedData.Put("ImageFamily", image.Family) s.GeneratedData.Put("ImageDescription", image.Description) s.GeneratedData.Put("ImageFolderID", image.FolderId) return multistep.ActionContinue } func (stepCreateImage) Cleanup(state multistep.StateBag) { // no cleanup }