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

134 lines
3.6 KiB
Go
Raw Normal View History

package vsphere_template
2017-07-09 14:12:37 -04:00
import (
"context"
"errors"
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/builder/vmware/iso"
2017-07-09 14:12:37 -04:00
"github.com/hashicorp/packer/common"
"github.com/hashicorp/packer/helper/config"
"github.com/hashicorp/packer/helper/multistep"
2017-07-09 14:12:37 -04:00
"github.com/hashicorp/packer/packer"
"github.com/hashicorp/packer/template/interpolate"
"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
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,
}
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
f := artifact.State(iso.ArtifactConfFormat)
k := artifact.State(iso.ArtifactConfKeepRegistered)
s := artifact.State(iso.ArtifactConfSkipExport)
if f != "" && k != "true" && s == "false" {
return nil, false, errors.New("To use this post-processor with exporting behavior you need set keep_registered as true")
}
// In some occasions the VM state is powered on and if we immediately try to mark as template
// (after the ESXi creates it) it will fail. If vSphere is given a few seconds this behavior doesn't reappear.
2017-08-24 21:46:35 -04:00
ui.Message("Waiting 10s for VMware vSphere to start")
2017-07-09 14:56:39 -04:00
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,
},
&stepMarkAsTemplate{
2017-08-30 17:19:54 -04:00
VMName: artifact.Id(),
2017-07-09 14:56:39 -04:00
},
}
2017-08-31 23:33:40 -04:00
runner := common.NewRunnerWithPauseFn(steps, p.config.PackerConfig, ui, state)
2017-07-09 14:56:39 -04:00
runner.Run(state)
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
}