packer-cn/post-processor/vsphere-template/post-processor.go

141 lines
3.4 KiB
Go
Raw Normal View History

package vsphere_template
2017-07-09 14:12:37 -04:00
import (
"context"
2017-07-09 14:12:37 -04:00
"fmt"
"net/url"
2017-07-09 20:33:03 -04:00
"strings"
2017-07-09 22:13:31 -04:00
"time"
2017-07-09 14:12:37 -04:00
"github.com/hashicorp/packer/common"
"github.com/hashicorp/packer/helper/config"
"github.com/hashicorp/packer/packer"
"github.com/hashicorp/packer/template/interpolate"
2017-07-09 14:56:39 -04:00
"github.com/mitchellh/multistep"
"github.com/vmware/govmomi"
2017-07-09 14:12:37 -04:00
)
2017-07-09 20:33:03 -04:00
var builtins = map[string]string{
"mitchellh.vmware-esx": "vmware",
}
2017-07-09 14:12:37 -04:00
type Config struct {
common.PackerConfig `mapstructure:",squash"`
Host string `mapstructure:"host"`
Insecure bool `mapstructure:"insecure"`
2017-07-09 14:12:37 -04:00
Username string `mapstructure:"username"`
Password string `mapstructure:"password"`
Datacenter string `mapstructure:"datacenter"`
2017-07-09 14:12:37 -04:00
VMName string `mapstructure:"vm_name"`
Folder string `mapstructure:"folder"`
ctx interpolate.Context
}
type PostProcessor struct {
config Config
url *url.URL
}
func (p *PostProcessor) Configure(raws ...interface{}) error {
err := config.Decode(&p.config, &config.DecodeOpts{
Interpolate: true,
InterpolateContext: &p.config.ctx,
InterpolateFilter: &interpolate.RenderFilter{
Exclude: []string{},
},
}, raws...)
if err != nil {
return err
}
errs := new(packer.MultiError)
vc := map[string]*string{
"host": &p.config.Host,
"username": &p.config.Username,
"password": &p.config.Password,
"vm_name": &p.config.VMName,
2017-07-09 14:12:37 -04:00
}
for key, ptr := range vc {
if *ptr == "" {
errs = packer.MultiErrorAppend(
errs, fmt.Errorf("%s must be set", key))
}
}
2017-07-18 23:10:05 -04:00
if p.config.Folder != "" && !strings.HasPrefix(p.config.Folder, "/") {
2017-07-09 14:56:39 -04:00
errs = packer.MultiErrorAppend(
2017-07-18 23:10:05 -04:00
errs, fmt.Errorf("Folder must be bound to the root"))
2017-07-09 14:56:39 -04:00
}
2017-07-18 23:10:05 -04:00
sdk, err := url.Parse(fmt.Sprintf("https://%v/sdk", p.config.Host))
if err != nil {
errs = packer.MultiErrorAppend(
errs, fmt.Errorf("Error invalid vSphere sdk endpoint: %s", err))
}
sdk.User = url.UserPassword(p.config.Username, p.config.Password)
p.url = sdk
2017-07-09 14:12:37 -04:00
if len(errs.Errors) > 0 {
return errs
}
return nil
}
func (p *PostProcessor) PostProcess(ui packer.Ui, artifact packer.Artifact) (packer.Artifact, bool, error) {
2017-07-09 20:33:03 -04:00
if _, ok := builtins[artifact.BuilderId()]; !ok {
return nil, false, fmt.Errorf("Unknown artifact type, can't build box: %s", artifact.BuilderId())
2017-07-09 20:33:03 -04:00
}
2017-07-09 14:56:39 -04:00
2017-07-09 20:33:03 -04:00
source := ""
for _, path := range artifact.Files() {
if strings.HasSuffix(path, ".vmx") {
source = path
break
}
}
2017-07-18 23:10:05 -04:00
// In some occasions when the VM is mark as template it loses its configuration if it's done immediately
// after the ESXi creates it. If vSphere is given a few seconds this behavior doesn't reappear.
2017-07-09 14:56:39 -04:00
ui.Message("Waiting 10s for VMWare vSphere to start")
time.Sleep(10 * time.Second)
2017-07-18 23:10:05 -04:00
c, err := govmomi.NewClient(context.Background(), p.url, p.config.Insecure)
if err != nil {
return nil, false, fmt.Errorf("Error connecting to vSphere: %s", err)
}
2017-07-31 10:30:13 -04:00
defer c.Logout(context.Background())
state := new(multistep.BasicStateBag)
state.Put("ui", ui)
state.Put("client", c)
2017-07-09 14:56:39 -04:00
steps := []multistep.Step{
2017-07-18 23:10:05 -04:00
&stepChooseDatacenter{
2017-07-09 14:56:39 -04:00
Datacenter: p.config.Datacenter,
},
&stepCreateFolder{
Folder: p.config.Folder,
},
2017-07-18 23:10:05 -04:00
&stepFetchVm{
2017-07-09 14:56:39 -04:00
VMName: p.config.VMName,
2017-07-09 20:33:03 -04:00
Source: source,
2017-07-09 14:56:39 -04:00
},
2017-07-18 23:10:05 -04:00
&stepMarkAsTemplate{},
&stepMoveTemplate{
2017-07-09 20:33:03 -04:00
Folder: p.config.Folder,
2017-07-09 14:56:39 -04:00
},
}
runner := &multistep.BasicRunner{Steps: steps}
runner.Run(state)
2017-07-09 14:12:37 -04:00
2017-07-09 14:56:39 -04:00
if rawErr, ok := state.GetOk("error"); ok {
return nil, false, rawErr.(error)
2017-07-09 14:56:39 -04:00
}
2017-07-09 14:12:37 -04:00
return artifact, true, nil
}