2014-12-29 06:46:54 -06:00
|
|
|
package common
|
|
|
|
|
|
|
|
import (
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
"os"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/mitchellh/multistep"
|
|
|
|
"github.com/mitchellh/packer/common"
|
|
|
|
commonssh "github.com/mitchellh/packer/common/ssh"
|
|
|
|
"github.com/mitchellh/packer/packer"
|
|
|
|
)
|
|
|
|
|
|
|
|
type CommonConfig struct {
|
2014-12-31 11:09:14 -06:00
|
|
|
Username string `mapstructure:"remote_username"`
|
|
|
|
Password string `mapstructure:"remote_password"`
|
|
|
|
HostIp string `mapstructure:"remote_host"`
|
2014-12-29 06:46:54 -06:00
|
|
|
|
|
|
|
VMName string `mapstructure:"vm_name"`
|
|
|
|
SrName string `mapstructure:"sr_name"`
|
|
|
|
FloppyFiles []string `mapstructure:"floppy_files"`
|
|
|
|
NetworkName string `mapstructure:"network_name"`
|
|
|
|
|
|
|
|
HostPortMin uint `mapstructure:"host_port_min"`
|
|
|
|
HostPortMax uint `mapstructure:"host_port_max"`
|
|
|
|
|
|
|
|
BootCommand []string `mapstructure:"boot_command"`
|
|
|
|
ShutdownCommand string `mapstructure:"shutdown_command"`
|
|
|
|
|
|
|
|
RawBootWait string `mapstructure:"boot_wait"`
|
|
|
|
BootWait time.Duration
|
|
|
|
|
|
|
|
ToolsIsoName string `mapstructure:"tools_iso_name"`
|
|
|
|
|
|
|
|
HTTPDir string `mapstructure:"http_directory"`
|
|
|
|
HTTPPortMin uint `mapstructure:"http_port_min"`
|
|
|
|
HTTPPortMax uint `mapstructure:"http_port_max"`
|
|
|
|
|
|
|
|
LocalIp string `mapstructure:"local_ip"`
|
|
|
|
|
|
|
|
// SSHHostPortMin uint `mapstructure:"ssh_host_port_min"`
|
|
|
|
// SSHHostPortMax uint `mapstructure:"ssh_host_port_max"`
|
|
|
|
SSHKeyPath string `mapstructure:"ssh_key_path"`
|
|
|
|
SSHPassword string `mapstructure:"ssh_password"`
|
|
|
|
// SSHPort uint `mapstructure:"ssh_port"`
|
|
|
|
SSHUser string `mapstructure:"ssh_username"`
|
|
|
|
RawSSHWaitTimeout string `mapstructure:"ssh_wait_timeout"`
|
|
|
|
SSHWaitTimeout time.Duration
|
|
|
|
|
|
|
|
OutputDir string `mapstructure:"output_directory"`
|
|
|
|
Format string `mapstructure:"format"`
|
|
|
|
KeepVM string `mapstructure:"keep_vm"`
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *CommonConfig) Prepare(t *packer.ConfigTemplate, pc *common.PackerConfig) []error {
|
|
|
|
var err error
|
|
|
|
|
|
|
|
// Set default values
|
|
|
|
|
|
|
|
if c.HostPortMin == 0 {
|
|
|
|
c.HostPortMin = 5900
|
|
|
|
}
|
|
|
|
|
|
|
|
if c.HostPortMax == 0 {
|
|
|
|
c.HostPortMax = 6000
|
|
|
|
}
|
|
|
|
|
|
|
|
if c.RawBootWait == "" {
|
|
|
|
c.RawBootWait = "5s"
|
|
|
|
}
|
|
|
|
|
|
|
|
if c.ToolsIsoName == "" {
|
|
|
|
c.ToolsIsoName = "xs-tools.iso"
|
|
|
|
}
|
|
|
|
|
|
|
|
if c.HTTPPortMin == 0 {
|
|
|
|
c.HTTPPortMin = 8000
|
|
|
|
}
|
|
|
|
|
|
|
|
if c.HTTPPortMax == 0 {
|
|
|
|
c.HTTPPortMax = 9000
|
|
|
|
}
|
|
|
|
|
|
|
|
if c.RawSSHWaitTimeout == "" {
|
|
|
|
c.RawSSHWaitTimeout = "200m"
|
|
|
|
}
|
|
|
|
|
|
|
|
if c.FloppyFiles == nil {
|
|
|
|
c.FloppyFiles = make([]string, 0)
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
if c.SSHHostPortMin == 0 {
|
|
|
|
c.SSHHostPortMin = 2222
|
|
|
|
}
|
|
|
|
|
|
|
|
if c.SSHHostPortMax == 0 {
|
|
|
|
c.SSHHostPortMax = 4444
|
|
|
|
}
|
|
|
|
|
|
|
|
if c.SSHPort == 0 {
|
|
|
|
c.SSHPort = 22
|
|
|
|
}
|
|
|
|
*/
|
|
|
|
|
|
|
|
if c.RawSSHWaitTimeout == "" {
|
|
|
|
c.RawSSHWaitTimeout = "20m"
|
|
|
|
}
|
|
|
|
|
|
|
|
if c.OutputDir == "" {
|
|
|
|
c.OutputDir = fmt.Sprintf("output-%s", pc.PackerBuildName)
|
|
|
|
}
|
|
|
|
|
2014-12-31 11:26:27 -06:00
|
|
|
if c.VMName == "" {
|
|
|
|
c.VMName = fmt.Sprintf("packer-%s-{{timestamp}}", pc.PackerBuildName)
|
|
|
|
}
|
|
|
|
|
2014-12-29 06:46:54 -06:00
|
|
|
if c.Format == "" {
|
|
|
|
c.Format = "xva"
|
|
|
|
}
|
|
|
|
|
|
|
|
if c.KeepVM == "" {
|
|
|
|
c.KeepVM = "never"
|
|
|
|
}
|
|
|
|
|
|
|
|
// Template substitution
|
|
|
|
|
|
|
|
templates := map[string]*string{
|
2014-12-31 11:09:14 -06:00
|
|
|
"remote_username": &c.Username,
|
|
|
|
"remote_password": &c.Password,
|
|
|
|
"remote_host": &c.HostIp,
|
2014-12-29 06:46:54 -06:00
|
|
|
"vm_name": &c.VMName,
|
|
|
|
"sr_name": &c.SrName,
|
|
|
|
"shutdown_command": &c.ShutdownCommand,
|
|
|
|
"boot_wait": &c.RawBootWait,
|
|
|
|
"tools_iso_name": &c.ToolsIsoName,
|
|
|
|
"http_directory": &c.HTTPDir,
|
|
|
|
"local_ip": &c.LocalIp,
|
|
|
|
"ssh_key_path": &c.SSHKeyPath,
|
|
|
|
"ssh_password": &c.SSHPassword,
|
|
|
|
"ssh_username": &c.SSHUser,
|
|
|
|
"ssh_wait_timeout": &c.RawSSHWaitTimeout,
|
|
|
|
"output_directory": &c.OutputDir,
|
|
|
|
"format": &c.Format,
|
|
|
|
"keep_vm": &c.KeepVM,
|
|
|
|
}
|
|
|
|
for i := range c.FloppyFiles {
|
|
|
|
templates[fmt.Sprintf("floppy_files[%d]", i)] = &c.FloppyFiles[i]
|
|
|
|
}
|
|
|
|
|
|
|
|
errs := make([]error, 0)
|
|
|
|
for n, ptr := range templates {
|
|
|
|
*ptr, err = t.Process(*ptr, nil)
|
|
|
|
if err != nil {
|
|
|
|
errs = append(errs, fmt.Errorf("Error processing %s: %s", n, err))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Validation
|
|
|
|
|
|
|
|
if c.Username == "" {
|
2014-12-31 11:09:14 -06:00
|
|
|
errs = append(errs, errors.New("remote_username must be specified."))
|
2014-12-29 06:46:54 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
if c.Password == "" {
|
2014-12-31 11:09:14 -06:00
|
|
|
errs = append(errs, errors.New("remote_password must be specified."))
|
2014-12-29 06:46:54 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
if c.HostIp == "" {
|
2014-12-31 11:09:14 -06:00
|
|
|
errs = append(errs, errors.New("remote_host must be specified."))
|
2014-12-29 06:46:54 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
if c.HostPortMin > c.HostPortMax {
|
|
|
|
errs = append(errs, errors.New("the host min port must be less than the max"))
|
|
|
|
}
|
|
|
|
|
|
|
|
if c.HTTPPortMin > c.HTTPPortMax {
|
|
|
|
errs = append(errs, errors.New("the HTTP min port must be less than the max"))
|
|
|
|
}
|
|
|
|
|
|
|
|
c.BootWait, err = time.ParseDuration(c.RawBootWait)
|
|
|
|
if err != nil {
|
|
|
|
errs = append(errs, fmt.Errorf("Failed to parse boot_wait: %s", err))
|
|
|
|
}
|
|
|
|
|
|
|
|
for i, command := range c.BootCommand {
|
|
|
|
if err := t.Validate(command); err != nil {
|
|
|
|
errs = append(errs,
|
|
|
|
fmt.Errorf("Error processing boot_command[%d]: %s", i, err))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if c.SSHKeyPath != "" {
|
|
|
|
if _, err := os.Stat(c.SSHKeyPath); err != nil {
|
|
|
|
errs = append(errs, fmt.Errorf("ssh_key_path is invalid: %s", err))
|
|
|
|
} else if _, err := commonssh.FileSigner(c.SSHKeyPath); err != nil {
|
|
|
|
errs = append(errs, fmt.Errorf("ssh_key_path is invalid: %s", err))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
if c.SSHHostPortMin > c.SSHHostPortMax {
|
|
|
|
errs = append(errs,
|
|
|
|
errors.New("ssh_host_port_min must be less than ssh_host_port_max"))
|
|
|
|
}
|
|
|
|
*/
|
|
|
|
|
|
|
|
if c.SSHUser == "" {
|
|
|
|
errs = append(errs, errors.New("An ssh_username must be specified."))
|
|
|
|
}
|
|
|
|
|
|
|
|
c.SSHWaitTimeout, err = time.ParseDuration(c.RawSSHWaitTimeout)
|
|
|
|
if err != nil {
|
|
|
|
errs = append(errs, fmt.Errorf("Failed to parse ssh_wait_timeout: %s", err))
|
|
|
|
}
|
|
|
|
|
|
|
|
switch c.Format {
|
|
|
|
case "xva", "vdi_raw":
|
|
|
|
default:
|
|
|
|
errs = append(errs, errors.New("format must be one of 'xva', 'vdi_raw'"))
|
|
|
|
}
|
|
|
|
|
|
|
|
switch c.KeepVM {
|
|
|
|
case "always", "never", "on_success":
|
|
|
|
default:
|
|
|
|
errs = append(errs, errors.New("keep_vm must be one of 'always', 'never', 'on_success'"))
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
if c.LocalIp == "" {
|
|
|
|
errs = append(errs, errors.New("A local IP visible to XenServer's mangement interface is required to serve files."))
|
|
|
|
}
|
|
|
|
*/
|
|
|
|
|
|
|
|
return errs
|
|
|
|
}
|
|
|
|
|
|
|
|
// steps should check config.ShouldKeepVM first before cleaning up the VM
|
|
|
|
func (c CommonConfig) ShouldKeepVM(state multistep.StateBag) bool {
|
|
|
|
switch c.KeepVM {
|
|
|
|
case "always":
|
|
|
|
return true
|
|
|
|
case "never":
|
|
|
|
return false
|
|
|
|
case "on_success":
|
|
|
|
// only keep instance if build was successful
|
|
|
|
_, cancelled := state.GetOk(multistep.StateCancelled)
|
|
|
|
_, halted := state.GetOk(multistep.StateHalted)
|
|
|
|
return !(cancelled || halted)
|
|
|
|
default:
|
|
|
|
panic(fmt.Sprintf("Unknown keep_vm value '%s'", c.KeepVM))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (config CommonConfig) GetSR(client XenAPIClient) (*SR, error) {
|
|
|
|
if config.SrName == "" {
|
|
|
|
// Find the default SR
|
|
|
|
return client.GetDefaultSR()
|
|
|
|
|
|
|
|
} else {
|
|
|
|
// Use the provided name label to find the SR to use
|
|
|
|
srs, err := client.GetSRByNameLabel(config.SrName)
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
switch {
|
|
|
|
case len(srs) == 0:
|
|
|
|
return nil, fmt.Errorf("Couldn't find a SR with the specified name-label '%s'", config.SrName)
|
|
|
|
case len(srs) > 1:
|
|
|
|
return nil, fmt.Errorf("Found more than one SR with the name '%s'. The name must be unique", config.SrName)
|
|
|
|
}
|
|
|
|
|
|
|
|
return srs[0], nil
|
|
|
|
}
|
|
|
|
}
|