126 lines
3.1 KiB
Go
126 lines
3.1 KiB
Go
//go:generate struct-markdown
|
|
//go:generate mapstructure-to-hcl2 -type CloneConfig
|
|
|
|
package clone
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
|
|
"github.com/hashicorp/packer/builder/vsphere/common"
|
|
"github.com/hashicorp/packer/builder/vsphere/driver"
|
|
"github.com/hashicorp/packer/helper/multistep"
|
|
"github.com/hashicorp/packer/packer"
|
|
)
|
|
|
|
type CloneConfig struct {
|
|
// Name of source VM. Path is optional.
|
|
Template string `mapstructure:"template"`
|
|
// The size of the disk in MB.
|
|
DiskSize int64 `mapstructure:"disk_size"`
|
|
// Create VM as a linked clone from latest snapshot. Defaults to `false`.
|
|
LinkedClone bool `mapstructure:"linked_clone"`
|
|
// Set network VM will be connected to.
|
|
Network string `mapstructure:"network"`
|
|
// VM notes.
|
|
Notes string `mapstructure:"notes"`
|
|
}
|
|
|
|
func (c *CloneConfig) Prepare() []error {
|
|
var errs []error
|
|
|
|
if c.Template == "" {
|
|
errs = append(errs, fmt.Errorf("'template' is required"))
|
|
}
|
|
|
|
if c.LinkedClone == true && c.DiskSize != 0 {
|
|
errs = append(errs, fmt.Errorf("'linked_clone' and 'disk_size' cannot be used together"))
|
|
}
|
|
|
|
return errs
|
|
}
|
|
|
|
type StepCloneVM struct {
|
|
Config *CloneConfig
|
|
Location *common.LocationConfig
|
|
Force bool
|
|
}
|
|
|
|
func (s *StepCloneVM) Run(ctx context.Context, state multistep.StateBag) multistep.StepAction {
|
|
ui := state.Get("ui").(packer.Ui)
|
|
d := state.Get("driver").(*driver.Driver)
|
|
|
|
vm, err := d.FindVM(s.Location.VMName)
|
|
|
|
if s.Force == false && err == nil {
|
|
state.Put("error", fmt.Errorf("%s already exists, you can use -force flag to destroy it", s.Location.VMName))
|
|
return multistep.ActionHalt
|
|
} else if s.Force == true && err == nil {
|
|
ui.Say(fmt.Sprintf("the vm/template %s already exists, but deleting it due to -force flag", s.Location.VMName))
|
|
err := vm.Destroy()
|
|
if err != nil {
|
|
state.Put("error", fmt.Errorf("error destroying %s: %v", s.Location.VMName, err))
|
|
}
|
|
}
|
|
|
|
ui.Say("Cloning VM...")
|
|
template, err := d.FindVM(s.Config.Template)
|
|
if err != nil {
|
|
state.Put("error", err)
|
|
return multistep.ActionHalt
|
|
}
|
|
|
|
vm, err = template.Clone(ctx, &driver.CloneConfig{
|
|
Name: s.Location.VMName,
|
|
Folder: s.Location.Folder,
|
|
Cluster: s.Location.Cluster,
|
|
Host: s.Location.Host,
|
|
ResourcePool: s.Location.ResourcePool,
|
|
Datastore: s.Location.Datastore,
|
|
LinkedClone: s.Config.LinkedClone,
|
|
Network: s.Config.Network,
|
|
Annotation: s.Config.Notes,
|
|
})
|
|
if err != nil {
|
|
state.Put("error", err)
|
|
return multistep.ActionHalt
|
|
}
|
|
if vm == nil {
|
|
return multistep.ActionHalt
|
|
}
|
|
state.Put("vm", vm)
|
|
|
|
if s.Config.DiskSize > 0 {
|
|
err = vm.ResizeDisk(s.Config.DiskSize)
|
|
if err != nil {
|
|
state.Put("error", err)
|
|
return multistep.ActionHalt
|
|
}
|
|
}
|
|
|
|
return multistep.ActionContinue
|
|
}
|
|
|
|
func (s *StepCloneVM) Cleanup(state multistep.StateBag) {
|
|
_, cancelled := state.GetOk(multistep.StateCancelled)
|
|
_, halted := state.GetOk(multistep.StateHalted)
|
|
if !cancelled && !halted {
|
|
return
|
|
}
|
|
|
|
ui := state.Get("ui").(packer.Ui)
|
|
|
|
st := state.Get("vm")
|
|
if st == nil {
|
|
return
|
|
}
|
|
vm := st.(*driver.VirtualMachine)
|
|
|
|
ui.Say("Destroying VM...")
|
|
|
|
err := vm.Destroy()
|
|
if err != nil {
|
|
ui.Error(err.Error())
|
|
}
|
|
}
|