//go:generate struct-markdown //go:generate mapstructure-to-hcl2 -type Config package ovf import ( "fmt" "github.com/hashicorp/packer-plugin-sdk/bootcommand" "github.com/hashicorp/packer-plugin-sdk/common" "github.com/hashicorp/packer-plugin-sdk/multistep/commonsteps" packersdk "github.com/hashicorp/packer-plugin-sdk/packer" "github.com/hashicorp/packer-plugin-sdk/template/config" "github.com/hashicorp/packer-plugin-sdk/template/interpolate" vboxcommon "github.com/hashicorp/packer/builder/virtualbox/common" ) // Config is the configuration structure for the builder. type Config struct { common.PackerConfig `mapstructure:",squash"` commonsteps.HTTPConfig `mapstructure:",squash"` commonsteps.FloppyConfig `mapstructure:",squash"` commonsteps.CDConfig `mapstructure:",squash"` bootcommand.BootConfig `mapstructure:",squash"` vboxcommon.ExportConfig `mapstructure:",squash"` vboxcommon.OutputConfig `mapstructure:",squash"` vboxcommon.RunConfig `mapstructure:",squash"` vboxcommon.CommConfig `mapstructure:",squash"` vboxcommon.ShutdownConfig `mapstructure:",squash"` vboxcommon.VBoxManageConfig `mapstructure:",squash"` vboxcommon.VBoxVersionConfig `mapstructure:",squash"` vboxcommon.GuestAdditionsConfig `mapstructure:",squash"` // The checksum for the source_path file. The type of the checksum is // specified within the checksum field as a prefix, ex: "md5:{$checksum}". // The type of the checksum can also be omitted and Packer will try to // infer it based on string length. Valid values are "none", "{$checksum}", // "md5:{$checksum}", "sha1:{$checksum}", "sha256:{$checksum}", // "sha512:{$checksum}" or "file:{$path}". Here is a list of valid checksum // values: // * md5:090992ba9fd140077b0661cb75f7ce13 // * 090992ba9fd140077b0661cb75f7ce13 // * sha1:ebfb681885ddf1234c18094a45bbeafd91467911 // * ebfb681885ddf1234c18094a45bbeafd91467911 // * sha256:ed363350696a726b7932db864dda019bd2017365c9e299627830f06954643f93 // * ed363350696a726b7932db864dda019bd2017365c9e299627830f06954643f93 // * file:http://releases.ubuntu.com/20.04/SHA256SUMS // * file:file://./local/path/file.sum // * file:./local/path/file.sum // * none // Although the checksum will not be verified when it is set to "none", // this is not recommended since these files can be very large and // corruption does happen from time to time. Checksum string `mapstructure:"checksum" required:"true"` // Additional flags to pass to // VBoxManage import. This can be used to add additional command-line flags // such as --eula-accept to accept a EULA in the OVF. ImportFlags []string `mapstructure:"import_flags" required:"false"` // Additional options to pass to the // VBoxManage import. This can be useful for passing keepallmacs or // keepnatmacs options for existing ovf images. ImportOpts string `mapstructure:"import_opts" required:"false"` // The filepath or URL to an OVF or OVA file that acts as the // source of this build. SourcePath string `mapstructure:"source_path" required:"true"` // The path where the OVA should be saved // after download. By default, it will go in the packer cache, with a hash of // the original filename as its name. TargetPath string `mapstructure:"target_path" required:"false"` // This is the name of the OVF file for the new virtual // machine, without the file extension. By default this is packer-BUILDNAME, // where "BUILDNAME" is the name of the build. VMName string `mapstructure:"vm_name" required:"false"` // Set this to true if you would like to keep // the VM registered with virtualbox. Defaults to false. KeepRegistered bool `mapstructure:"keep_registered" required:"false"` // Defaults to false. When enabled, Packer will // not export the VM. Useful if the build output is not the resultant image, // but created inside the VM. SkipExport bool `mapstructure:"skip_export" required:"false"` ctx interpolate.Context } func (c *Config) Prepare(raws ...interface{}) ([]string, error) { err := config.Decode(c, &config.DecodeOpts{ PluginType: vboxcommon.BuilderId, // "mitchellh.virtualbox" Interpolate: true, InterpolateContext: &c.ctx, InterpolateFilter: &interpolate.RenderFilter{ Exclude: []string{ "boot_command", "guest_additions_path", "guest_additions_url", "vboxmanage", "vboxmanage_post", }, }, }, raws...) if err != nil { return nil, err } // Defaults if c.VMName == "" { c.VMName = fmt.Sprintf( "packer-%s-%d", c.PackerBuildName, interpolate.InitTime.Unix()) } // Prepare the errors var errs *packersdk.MultiError errs = packersdk.MultiErrorAppend(errs, c.ExportConfig.Prepare(&c.ctx)...) errs = packersdk.MultiErrorAppend(errs, c.ExportConfig.Prepare(&c.ctx)...) errs = packersdk.MultiErrorAppend(errs, c.FloppyConfig.Prepare(&c.ctx)...) errs = packersdk.MultiErrorAppend(errs, c.CDConfig.Prepare(&c.ctx)...) errs = packersdk.MultiErrorAppend(errs, c.HTTPConfig.Prepare(&c.ctx)...) errs = packersdk.MultiErrorAppend(errs, c.OutputConfig.Prepare(&c.ctx, &c.PackerConfig)...) errs = packersdk.MultiErrorAppend(errs, c.RunConfig.Prepare(&c.ctx)...) errs = packersdk.MultiErrorAppend(errs, c.ShutdownConfig.Prepare(&c.ctx)...) errs = packersdk.MultiErrorAppend(errs, c.CommConfig.Prepare(&c.ctx)...) errs = packersdk.MultiErrorAppend(errs, c.VBoxManageConfig.Prepare(&c.ctx)...) errs = packersdk.MultiErrorAppend(errs, c.VBoxVersionConfig.Prepare(c.CommConfig.Comm.Type)...) errs = packersdk.MultiErrorAppend(errs, c.BootConfig.Prepare(&c.ctx)...) errs = packersdk.MultiErrorAppend(errs, c.GuestAdditionsConfig.Prepare(c.CommConfig.Comm.Type)...) if c.SourcePath == "" { errs = packersdk.MultiErrorAppend(errs, fmt.Errorf("source_path is required")) } if c.GuestAdditionsInterface == "" { c.GuestAdditionsInterface = "ide" } // Warnings var warnings []string if c.ShutdownCommand == "" { warnings = append(warnings, "A shutdown_command was not specified. Without a shutdown command, Packer\n"+ "will forcibly halt the virtual machine, which may result in data loss.") } // Check for any errors. if errs != nil && len(errs.Errors) > 0 { return warnings, errs } // TODO: Write a packer fix and just remove import_opts if c.ImportOpts != "" { c.ImportFlags = append(c.ImportFlags, "--options", c.ImportOpts) } return warnings, nil }